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第 1 章 项 目 需求 分 析 与 设计 


本 章 学 习 要 点 : 

。 了 解 当前 主流 Web 项 目 开发 技术 及 其 特点 ; 
。 掌握 与 客户 沟通 的 能 力 ; 

。 熟练 掌握 项 目 需 求 分 析 报告 的 编写 规范 。 


1.1 开发 技术 的 选取 


近年 来 ,我 国 软件 产业 发 展 始终 保持 较 快 的 增长 。 基 于 Web 的 应 用 软件 由 于 具有 投 
入 成 本 比较 小 ,软件 开发 比较 容易 .对 企业 快速 扩张 支持 较 好 等 特点 ,成 为 软件 开发 领域 
的 发 展 趋势 。 目 前 主流 的 Web 开发 技术 有 如 下 几 种 。 

(1) PHP。PHP 是 英文 超级 文本 预 处 理 语 言 Hypertext Preprocessor 的 缩写 。PHP 
是 一 种 HTML 内 和 骸 式 的 语言 ,是 一 种 在 服务 器 端 执 行 的 嵌入 HTML 文档 的 脚本 语言 ， 
语言 的 风格 类 似 于 C 语言 ,被 广泛 地 运用 。 

(2) dotNET。dotNET 就 是 . NET, 严 格 地 说 是 . Net Framework 框架 .. NET 是 一 
个 微软 开发 的 编程 环境 ,可 以 使 用 ASP、C#(# 读 作 sharp)、VB 等 多 种 编程 语言 。 

.NET 是 Microsoft 用 于 创建 XML Web 服务 (下 一 代 软 件 ) 的 平台 ,借助 于 该 平台 ， 
可 以 创建 和 使 用 基于 XML 的 应 用 程序 、 进 程 和 Web 站 点 以 及 服务 。 借 助 于 微软 强大 的 
推广 能 力 和 快速 的 版 本 更 新 ,. NET 已 经 迅速 成 为 主流 的 Web 开发 技术 。 

(3) Java Web 技术 。 以 Java 为 基础 的 Web 开发 技术 .统称 为 Java Web 技术 。 由 于 
Java 语言 本 身 的 安全 性 设计 ,使 得 Java Web 的 安全 性 较 好 。 本 书 中 的 中 小 型 Web 项 目 
的 开发 就 是 采用 了 Java Web 技术 。 

这 三 种 开发 技术 没有 严格 的 “ 优 ”与 “ 劣 ”之 分 ,Web 项 目的 开发 到 底 采 用 何 种 技术 ， 
是 项 目的 规模 ,安全 性 要 求 .预算 甚至 开发 人 员 的 使 用 习惯 等 因素 权衡 之 下 的 考量 。 

TIOBE 编程 语言 社区 排行 榜 是 编程 语言 流行 趋势 的 一 个 指标 ,每 月 更 新 。 排 行 榜 排 
名 基于 互联 网 上 有 经 验 的 程序 员 .课程 和 第 三 方 厂商 的 数量 。 排 名 使 用 著名 的 搜索 引擎 
(诸如 Google、MSN 雅虎 ) 以 及 Wikipedia 和 YouTube 进行 计算 。 

表 1-1 是 2013 年 5 月 编程 语言 的 排行 .可 以 看 到 三 种 主流 的 Web 开发 技术 在 语言 
使 用 排行 榜 上 均 排名 靠 前 。 
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表 1-1 TIOBE 编程 语言 排行 


Position Position a Ratings Delta 

May 2013 | May 2012 Delta in Position |Programming Language May 2013 | May 2012 Status 
1 1 一 C 18.729% | +1.38% A 
2 , = Java 16.914% | +0.31% A 
3 4 全 Objective-C 10.428% | 十 2.12%% A 
4 3 § C+#H+ 9.198% —0.63% A 
5 5 = C# 6.119% —0.70% A 
6 6 = PHP 5.784% 十 0.07% A 


读者 可 以 通过 http://www. tiobe. com/index. php/content/paperinfo/tpci/index. 
html 网 址 查看 最 新 的 TIOBE 排名 。 


1.2 Java Web 开发 模式 


根据 所 开发 的 项 目的 规模 ,可 采用 以 下 不 同 的 Java Web 开发 模式 。 

(1) 直接 使 用 JSP。 对 于 小 型 Web 站 点 ,可 以 直接 使 用 JSP 来 构建 动态 网 站 。 

(2) JSP+JavaBeans。 在 该 模式 中 ,JSP 页 面 独自 响应 请 求 并 将 处 理 结果 返回 客户 。 
所 有 的 数据 通过 Bean 来 处 理 ,JSP 实现 页 面 的 表现 。 该 模式 也 实现 了 页 面 的 表现 和 页 面 
商业 人 逻辑 的 分 离 ,可 以 很 好 地 满足 小 型 应 用 的 需要 。 


(3) JSP 十 JavaBeans 十 Servlet。 这 种 模式 的 主要 思想 是 使 用 一 个 或 多 个 Servlet 作 
为 控制 器 。 多 个 Servlet 控制 器 可 以 结合 起 来 完成 复杂 的 任务 ,可 重用 性 好 ,可 用 作 中 小 
型 项 目的 开发 模式 。 


(4) Java EE 开发 模式 。Java EE 基于 标准 的 体系 结构 和 组 件 开发 ,采用 了 松散 的 设 
计 方 法 ,组 件 既 可 以 单独 调用 ,也 可 以 组 合 调 用 。Java EE 组 件 解 决 了 所 有 底层 复杂 的 问 
题 , 组 件 易 于 升级 。 组 件 和 门户 基于 XML 配置 的 方式 ,可 以 灵活 配置 。 能 提供 良好 的 可 
开发 系统 外 部 接口 ,适合 开发 中 到 大 型 网 站 项 目 。 

本 书 将 带领 读者 一 同 为 “ 佳 衣 屋 ”公司 开发 一 个 基于 JSP 十 JavaBeans 十 Servlet 技术 
的 中 型 信息 系统 ,在 软件 项 目 进 入 项 目 开 发 之 前 要 进行 一 系列 的 准备 工作 ,包括 与 用 户 充 
分 沟通 并 进行 需求 分 析 、 搭 建 开 发 和 测试 环境 以 及 设计 数据 库 等 工作 。 


1.3 软件 项 目的 需求 分 


按照 软件 产品 开发 过 程 的 技术 特点 ,可 将 软件 分 为 定制 软件 (customized software) 
和 通用 软件 (packaged software) 。 其 中 ,定制 软件 是 按照 单个 客户 的 个 性 化 要 求 , 以 软件 
项 目的 方式 为 其 提交 个 性 化 的 解决 方案 。 通 用 软件 是 以 通用 软件 包 的 方式 提交 给 不 同 的 
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用 户 来 使 用 。Web 项 目 多 为 定制 软件 ,需要 根据 用 户 的 具体 情况 .具体 要 求 进行 设计 ,并 
提供 相应 的 服务 。 

需求 分 析 是 定制 Web 项 目 开发 的 第 一 步 , 也 是 非常 关键 的 一 步 。 需 求 分 析 是 指 开 发 
人 员 与 用 户 进行 沟通 ,理解 用 户 需 求 ,就 软件 功能 与 客户 达成 一 致 ,最 终 形成 开发 计划 的 
一 个 过 程 。 需 求 分 析 的 结果 可 以 形成 (需求 分 析 报 告 》。 用 户 确认 并 签署 (需求 分 析 报 告 》 
后 ,项 目 就 可 进入 开发 阶段 了 。 


1.4 需求 分 析 报 告 的 格式 


需求 分 析 报 告 因 所 开发 的 软件 项 目 类 型 的 不 同 会 有 所 区 别 , 本 节 介 绍 中 小 型 Web 项 
目的 需求 分 析 报告 的 常用 格式 。 


1.4.1 编写 目的 


需求 分 析 报 告 的 引言 部 分 一 般 是 阐述 项 目 开 发 的 目的 和 意义 ,使 阅读 者 对 项 目 有 一 
个 整体 的 了 解 。 

关 示 例 1-1 “ 佳 衣 屋 "项 目 需求 分 析 报告 中 的 编写 目的 

“ 佳 衣 屋 "信息 系统 需求 分 析 报告 

1 编写 目的 

本 报告 的 目的 是 根据 客户 的 需求 ,对 系统 功能 、 性 能 需求 向 * 佳 衣 屋 公司、 项 目 组 开 
发 成 员 、 项 目 实施 组 和 测试 成 员 提 供 一 个 清晰 的 陈述 ,为 项 目的 后 续 展 开 提 供 指导 。 

本 报告 主要 曾 述 “ 佳 衣 屋 ”公司 服装 连锁 销售 的 进 、 销 \ 存 业务 流程 ,兼顾 员工 的 管理 
和 报表 分 析 等 功能 。 项 目 上 线 运行 后 ,信息 系统 的 使 用 者 可 以 及 时 了 解 商品 的 销售 .库存 
等 信息 ,并 能 够 及 时 地 对 商品 进行 调度 ,有 助 于 公司 的 运营 和 管理 。 


1.4.2 运行 环境 


这 一 部 分 说 明 项 目的 开发 和 运行 所 需 的 软 、 硬 件 条 件 。 
关 示 例 1-2 “ 佳 衣 屋 "项 目 需求 分 析 报告 中 的 运行 环境 
2 运行 环境 

硬件 : 机 架 式 服务 器 

操作 系统 : Windows Server 2003 

支撑 环境 : JDK1.6,MySQL 5.1,Tomcat 7.0 

其 他 相关 软件 : MyEclipse 9.0,SQLyog 9.0 


中 小 型 Web 项 目 开 发 实战 


1.4.3 系统 结构 分 析 


一 个 软件 系统 是 由 许多 功能 模块 构成 的 。 模 块 化 是 一 种 重要 的 设计 思想 ,这 种 思想 
把 一 个 复杂 的 系统 分 解 为 一 些 规模 较 小 、 功 能 较 简单 ,更 易于 建立 和 修改 的 部 分 ,各 个 模 
块 具有 相对 独立 性 ,可 以 分 别 加 以 设计 实现 。 

系统 结构 通常 以 树 形 框图 即 功能 结构 图 的 形式 来 描述 模块 之 间 的 关系 和 相互 作用 ， 
图 中 的 每 一 个 框 代表 一 个 功能 模块 。 每 个 功能 模块 还 可 以 继续 分 解 为 第 三 层 、 第 四 层 其 
至 更 多 的 子 功能 模块 。 

将 示 例 1-3 “ 佳 衣 屋 "项 目的 系统 功能 结构 图 如 图 1-1 所 示 

3 系统 结构 分 析 


添加 员工 信息 
员工 管理 “| | “| 修改 员工 信息 
删除 员工 信息 
一 | 登录 
添加 商品 信息 
失败 | 商品 信息 管理 修改 商品 信息 
身份 验证 一 删除 商品 信息 
| 一 | 报表 分 析 
成 功 s 
查看 销售 记录 
响应 
权限 识别 响应 分 店 的 调 货 请 求 
i 添加 本 店 库存 信息 
一 一 一 本 店 库存 管理 | | 修改 本 店 库存 信息 
| [| 删除 本 店 库存 信息 
添加 本 店 销售 信息 
本 让 销售 从 昌吉 趟 店铺 人 信和 
分 店 店员 oe EE 
向 总 店 调 货 


图 1-1 “ 佳 衣 屋 ” 项 目 需 求 分 析 中 的 功能 结构 图 


1.4.4 系统 功能 分 析 


系统 功能 分 析 部 分 是 把 功能 模块 做 一 个 比较 详细 的 说 明 ,用 来 指导 编程 实现 。 
闪 示 例 1-4 “ 佳 衣 屋 ”项 目 需求 分 析 报告 中 的 系统 功能 分 析 部 分 
4 系统 功能 分 析 


4.1 登录 模块 
用 户 提交 用 户 名 、 密 码 、 验 证 码 .选择 权限 (管理 员 / 总 店 店员 /分 店 店员 ) 选择 是 否 记 
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住 密码 , 单 击 “ 登 录 ? 按 钮 进行 身份 验证 , 单 击 * 重 置 ?按钮 则 清空 用 户 输入 的 信息 。 用 户 界 
面 如 图 1-2 所 示 。 


网 站 Logo 
用 户 名 : 
密 码 : 
验证 码 : 
权 限 :_ (列表 菜单 ) 
是 否 记 住 密码 : (列表 菜单 ) 
登录 | | 重 置 


图 1-2 “ 佳 衣 屋 ” 项 目 登 录 页 示意 图 


( 略 ) 

4.2 管理 员 权限 下 的 功能 模块 

4.2.1 商品 管理 模块 

1. 添加 商品 子 模块 

管理 员 单 击 左 侧 功 能 导航 栏 中 的 “商品 管理 ”下 的 “添加 商品 信息 ” 超 链接 ,在 右 侧 栏 
中 出 现 填写 商品 信息 的 表单 ,输入 商品 的 款 号 ,名 称 、 颜 色 、 销 售 指导 价 并 上 传 商品 图 片 ， 
单 击 “ 添 加 ”按钮 后 ,将 新 商品 的 基本 信息 添加 到 系统 中 。 用 户 界 面 如 图 1-3 所 示 。 


网 站 Logo 

员工 管理 款 号 : 

商品 管理 名 称 : 

查询 商品 信息 | 颜 色 : 

添加 商品 信息 | 图 片 : 

查看 销售 情况 。 | 销售 指导 价 : 
报表 分 析 

添加 | | 重 置 


图 1-3 “ 佳 衣 屋 ”项 日 管理 员 添 加 商品 页 示意 图 


( 略 ) 
4.3 分 店 店员 权限 下 的 功能 模块 
( 略 ) 

4.3.3 进货 模块 


分 店 店员 从 总 店 库存 商品 列表 中 选中 需要 调 人 的 商品 ,并 填写 每 种 商品 的 进货 量 , 单 
击 “ 进 货 ” 按 钮 后 完成 批量 进货 请 求 的 发 送 。 用 户 界面 如 图 1-4 所 示 。 


中 小 型 Web 项 目 开 发 实战 


网 站 Logo 
库存 管理 款 号 名 称 类 型 码 数 颜色 售 价 ”库存 量 进货 量 
销售 管理 2 

yy I NY IY bp ft 
进货 查询 2 

进货 


图 1-4 分 店 店员 从 总 店 进货 页 示意 图 


( 略 ) 

4.4 总 店 店员 权限 下 的 功能 模块 

4.4.1 进货 请 求 处 理 模块 

总 店 店员 单 击 左 侧 功能 导航 栏 中 的 “发 货 ? 超 链接 ,在 右 侧 栏 中 出 现 各 分 店 的 进货 请 
求 及 当前 处 理 的 阶段 。 

总 店 店员 单 击 操 作 栏 中 的 “确认 ”按钮 ,表示 对 进货 请 求 予以 确认 。 如 果 库 存 数量 大 
于 进货 数量 , 则 该 条 进货 请 求 的 操作 栏 出 现 “ 发 货 ” 按 钮 ,分 店 的 该 条 进货 请 求 状态 将 变 为 
“ 待 发 货 ”; 如果 库存 数量 小 于 进货 数量 , 则 提示 “库存 不 足 ”, 总 店 店员 可 单 击 * 取 消 ?按钮 
来 取消 此 次 请 求 ,总 店 和 分 店 的 该 条 进货 请 求 状 态 将 变 为 “已 取消 ”, 或 者 待 库存 增加 后 再 
确认 此 进货 请 求 。 

总 店 店员 单 击 操作 栏 中 的 “发 货 " 按 钮 ,表示 已 将 商品 发 往 分 店 , 总 店 和 分 店 的 该 条 进 
货 请 求 状态 将 变 为 “已 发 货 *"。 用 户 界面 如 图 1-5 所 示 。 


网 站 Logo 


库存 管理 款 号 颜色 类 型 尺码 数量 订购 日 期 处 理 日 期 分 店 操作 
销售 管理 XXX XXX XX XX XX XX-XX-XX XX-XX-XX XX 确认 | | 取消 


进 货 TY WY I I SII YI 发 货 


ZZZ ZZZZ ZZ 77 277 772-272-272 V2-27-7Z 227 已 发 货 


图 1-5 分 店 店员 从 总 店 进货 页 示意 图 


因为 篇 幅 的 限制 ,本章 中 的 需求 分 析 及 本 书后 面 的 章节 中 的 示例 , 仅 能 包含 一 个 真实 
连锁 销售 进 、 销 、 存 管理 系统 中 的 部 分 功能 ,希望 随 着 章节 的 推进 ,读者 能 够 学 以 致 用 , 充 
分 发 挥 主动 性 ,将 这 一 系统 完善 。 
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课 后 练 习 


1. 读者 可 以 为 * 佳 衣 屋 ?项目 添 加 功能 模块 ,并 在 系统 结构 图 中 画 出 。 
2. 请 自由 组 织 起 来 到 企 事 业 单位 进行 实地 考察 ,了 解 它们 的 经 营 或 者 管理 流程 ,以 
及 信息 化 需求 , 写 出 需求 分 析 报 告 。 


第 2 章 MyEclipse 集成 开发 环境 


本 章 学 习 要 点 : 
。 熟练 掌握 MyEclipse 集成 开发 环境 (IDE) 中 创建 Web 项 目的 方法 ; 
。 熟练 掌握 MyEclipse 集成 开发 环境 中 项 目的 调试 方法 。 


Java Web 项 目的 开发 和 运行 ,需要 安装 必要 的 软件 ,并 进行 一 定 的 配置 。 
2.1 任务 一 : JDK 的 安装 和 配置 


2.1.1 JDK 的 下 载 和 安装 


学 习 Java 语言 的 读者 对 JDK (Java Development Kit) 一 定 不 陌生 ,JDK 包含 了 多 个 
用 于 Java 开发 的 组 件 , 其 中 包括 以 下 工具 。 

。javac: 编译 器 ,将 后 级 名 为 .java 的 源 代码 编译 成 后 组 名 为 . class 的 字 节 码 。 

。 Java: 运行 工具 ,运行 . class 的 字 节 码 。 

。jar: 打包 工具 ,将 相关 的 类 文件 打包 成 一 个 文件 。 

。javadoc: 文档 生成 器 ,从 源码 注释 中 提取 文档 ,注释 需 符合 规范 。 

。 jdb debugger: 调试 工具 。 

JDK 中 还 包括 完整 的 JRE(Java Runtime Environment,Java 运行 环境 ) ,也 被 称 为 
private runtime, 它 包括 了 用 于 产品 环境 的 各 种 库 类 ,如 基础 类 库 rt. jar, 以 及 给 开发 人 员 
使 用 的 补充 库 , 如 国际 化 与 本 地 化 的 类 库 、IDL 库 等 。 

JDK 中 还 包括 各 种 样 例 程序 ,用 以 展示 Java API 中 的 各 个 部 分 。 

可 以 从 Oracle 的 官方 网 站 (http://www. oracle. com/ ) 上 搜索 并 下 载 最 新 的 适用 于 
Windows 平 台 的 JDK 版 本 ,如 图 2-1 所 示 , 并 按照 安装 向 导 的 提示 进行 安装 即 可 。 


2.1.2 JDK 环境 变量 的 设置 


JDK 安装 完成 后 ,还 要 进行 环境 变量 的 设置 。 步 又 如 下 。 
(1) 单 击 系统 的 “开始 "菜单 ,打开 “控制 面板 ”双击 “系统 "工具 , 单 击 任务 中 的 “高 级 
系统 设置 ”, 如 图 2-2 所 示 。 


76.93 MB 各 jd-5u29inuei586-Tpm bin 


选择 Windows 版 本 
的 JDK 。Oracle 网 
站 总 在 更 新 ， 读 者 | 
看 到 的 网 站 内 容 可 一 上 一 亏 
能 会 与 此 图 不 同 | 


图 2-1 从 Oracle 网 站 下 载 JDK 


| _ 单 击 任务 栏 中 的 
“高 级 系统 设置 ” 


图 2-2 “系统 "工具 中 的 任务 栏 


(2) 单 击 弹出 的 “系统 属性 ”对 话 框 中 的 “高 级 "选项 卡 ,之 后 单 击 该 选项 卡 中 的 “环境 
变量 ?按钮 ,可 以 对 环境 变量 进行 查看 ,如 图 2-3 所 示 。 


| 


计算 机 名 | 硬件 。 高 级 “| 系统 保护 | 运程 | 
要 进行 大 多 下 更 次 ， 您 必须 作为 管理 员 登 录 。 


"性 说 
视觉 效果 ， 处 理 器 计划 ， 内 存 使 用 ， 以 及 虚拟 内 存 


LS) 


三 用 户 配置 文件 
与 您 登录 有 关 的 点 面 设置 


三 启动 和 部 障 恢复 
系统 启动 、 系 统 失败 和 i 同 试 信息 


单 击 “ 环 境 变量 "按钮 


2-3 “系统 属性 "对话 框 
11 


(3) 单 击 * 环 境 变量 ”对话 框 下 方 的 “系统 变量 "区域 中 的 "新建 ?按钮 ,进入 环境 变量 
的 编辑 状态 ,如 图 2-4 所 示 。 


CrWWindowsNsysten32Vend exe 
FALSE 
加 单 击 “ 新 建 "按钮 
2 \Progran Files\Java\jdkl. 6.0, 
MMRRR NF_PR 


图 2-4 “环境 变量 "对 话 框 
(4) tga java_home;“ 变 量 值 ”文本 


框 中 输入 JDK 所 在 的 目录 , 单 击 “ 确 定 ” 按 钮 。 图 2-5 所 示 的 例子 是 将 JDK 1. 6 安装 在 
D 盘 的 \Program Files 目录 下 。 ee et 


(1) 输入 变量 名 java_home 
(2) 输入 JDK 的 安装 目录 
(3) 单 击 “ 确 定 "按钮 


图 2-5 设置 系统 变量 java_home 


(5) 检查 一 下 目前 系统 有 无 名 叫 classpath 的 变量 ,如 果 有 , 则 选中 它 并 单 击 “ 编 辑 ” 
按钮 ,如 图 2-6 所 示 。 在 弹出 的 “编辑 系统 变量 ”对 话 框 中 .添加 变量 值 *; %java_home%\ 
lib; %java_home%\lib\tools.jar”, 应 注意 不 同 的 变量 间 应 用 分 号 隔 开 ,之 后 单 击 “ Me 
按钮 ,如 图 2-7 所 示 ; 如 果 系 统 当 前 没有 classpath 变量 ,. 则 需要 按照 步骤 (4) 新 建 一 
classpath 变量 。 

(6) 单 击 系统 变量 中 的 path 变量 , 单 击 “编辑 ?按钮 ,如 图 2-8 所 示 。 在 弹出 的 “编辑 
系统 变量 ”对 话 框 中 ,添加 “ %java_home%\bin;%java_home%\jre6\bin”, 单 击 “ 确 定 ” 按 
钮 ,如 图 2-9 所 示 。 

设置 完 环境 变量 后 ,进入 命令 行 模式 ,输入 命令 java -version。 如 果 命 令 得 到 响应 ， 
系统 会 输出 JDK 的 版 本 号 ,如 图 2-10 所 示 , 则 表明 JDK 的 安装 和 环境 的 设置 已 经 正确 完 
万 下。 
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TENP WUSERFROFILEX\AppData\Local \Tenp 
TIP XUSERPROPILEX\AppData\Local \Tenp 


(1) 选中 classpath 变 量 


CO) 单 击 "编辑 "按钮 win 
I _ 量 除 0 


编辑 系统 变量 


(1) 添加 JDK 的 jar 包 


(2) 单 击 "确定 "按钮 


WUSERPROFILEX\AppData\Local\Temp 
WUSERPROFILEX\AppData\Local\Temp 


新 建 和 )... | 编辑 EE)... 肿 除 O) 


COW: .EXE:. BAT;. CMD:. YVES; . VBE 


2 | 
强 诗 .| 编 站 | WW | 


| 


图 2-8 编辑 系统 变量 Path 
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Path 


%java_homk%\bin: %java_home%\ jreB Ubi 


图 2-9 将 JDK 的 可 执行 文件 目录 添加 到 系统 的 Path 变量 中 


JavaCIM> SE Runtime Environnent build 1.6.9 24-b87> 
lJava HotSpot CIM> Client UM build 19.1-b82, mixed mode, sharing) 


图 2-10 检验 JDK 的 安装 与 环境 变量 的 设置 


2.2 任务 二 : Tomcat 的 安装 和 配置 


Tomcat 是 目前 比较 流行 的 Web 应 用 服务 器 之 一 ,是 一 个 开源 项 目 ,运行 时 占用 的 系 
统 资源 小 ,扩展 性 好 ,支持 负载 平衡 与 邮件 服务 等 开发 应 用 系统 常用 的 功能 。 

Tomcat 是 一 个 轻 量 级 的 应 用 服务 器 ,在 中 小 型 系统 和 并 发 访问 用 户 不 是 很 多 的 场 
合 下 被 普遍 使 用 ,是 开发 和 调试 Java Web 项 目的 首选 。 


”任务 目标 
: 下 载 并 安装 Tomcat 服务 器 。 安 装 完成 后 启动 服务 器 ,在 浏览 器 地 址 栏 中 输入 正 
; 确 的 地 址 ,可 以 看 到 Tomcat 默认 的 主页 。 


Tomcat 的 最 新 版 本 可 以 从 Apache 的 网 站 上 下 载 , 如 图 2-11 所 示 。 


EN repomat phe modon 
无 病 


下 一 个 “上 一 个 “ 忆 高 让 关闭 


7.0.22 
Please see the README file for packaging information It explains what every distribution contains 
[nevpatibuiom | 
选择 并 下 载 Windows » Core 
版 本 的 Tomcat o zp (pgp. md5) 
Apache 网 站 总 在 更 新 ， 
大 家 看 到 的 网 站 内 容 


-bi Windows zip (pgp. md5) 
-bi Windows zip (pgp.md5 


可 能 会 与 此 图 不 一 定 o 64-bat haniom Windows zip (pgp, md5) 
相同 © 32-bi/64-bit Windows Service Installer (pgp, md5) 


图 2-11 从 Apache 网 站 下 载 Tomcat 7.0 
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双击 下 载 的 安装 软件 进行 安装 。Tomcat 的 安装 很 简单 ,可 以 按照 安装 向 导 的 指引 
完成 ,需要 注意 的 是 配置 选项 页 的 设置 ,如 图 2-12 所 示 。Tomcat 默认 的 端口 号 是 8080， 
当 服 务 器 上 的 8080 端口 已 被 占用 时 ,这 个 端口 号 需要 被 修改 ,以 免 造 成 端口 冲突 。 


四 =|Gjx| 
Configuration 
Tomcat basic configuration 
Server Shutdown Port [evos 
设置 Tomcat 的 端 
HTTP/L 1 Connector Port [eos0 口号 ， 默 认为 8080 
AJP/1.3 Connector Port Teoos 
Windows Service Name Fromcat7 
Create shorteuts fo dl users Cc 
Tomcat Administrator Logn 。 User Name |[ 
(optional) 
Password | 
Roles [manaoer gs 
Nullsoft Install System v2,46 
<Back Next > Cancel | 


图 2-12 ” Tomcat 7.0 的 安装 


安装 完成 后 , 单 击 系统 的 “开始 "菜单 ,找到 Apach Tomcat 7.0 选项 ,如 图 2-13 所 示 ， 
单 击 该 选项 下 的 Configure Tomcat, 弹 出 如 图 2-14 所 示 的 对 话 框 。 启 动 Tomcat。 


General | Log on| Logeine | Juva | startup | Shuatdorn| 

(1) 单 击 Start 按 钮 ， 

当 看 到 Star 近 包 交 | 下 站 
灰 、Stop 按 钮 变 亮 
时 ， 表 明 服务 器 ~ [poche Tomcat 7.0.11 Server -htip://tomcat.apache,, 
已 经 启动 Paih to executable: 

[SProgan Fies\omcat 7.0pnWomcat7 exe" /RS//Tomcat7 
| 


有 园 Apache Tomcat 7.0 Tomcat7 


nfigure Tomcat 


Monitor Tomca 
Start Pause Restat 


加 Tomcat 7.0 Program Directory 


Service Status: «Stopped 


加 Tomcat Documentation 


口 Tomcat Home page (2) 单 击 "确定 "按钮 一 NE | 3 | ER | 


图 2-13 Tomcat 服务 器 的 启动 程序 图 2-14 启动 Tomcat 服务 器 


待 服务 器 正常 启动 后 ,打开 IE 或 其 他 浏览 器 ,在 地 址 栏 中 输入 http://localhost: 
8080 ,如 果 看 到 如 图 2-15 所 示 的 页 面 , 则 表明 Tomcat 运行 正常 。 

localhost 是 本 地 主机 的 意思 ,表明 服务 器 就 是 在 当前 使 用 的 这 台 机 器 上 。 如 果 是 进 
行 远 程 的 访问 ,可 以 用 服务 器 的 IP 地 址 来 替换 localhost; 如 果 为 服务 器 配置 了 域名 ,就 可 
以 通过 域名 来 访问 了 。 

在 文件 名 默认 的 情况 下 ,服务 器 会 自动 地 寻找 根 目录 下 的 index. html\index. htm 或 
者 index. jsp 文件 等 默认 的 主页 。 

梅 
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] FT- 人 上 gm A 


Home Documentation “Configuraton “EXamples 一 Wi Mailing Lists Find Help 
Apache Tomcat/7.0.22 


“eA he Software Foundation 


http://www.apache 


Ifyou're seeing this, you've successfully installed Tomcat Congratulations! 


™ Recommended Reading: 


| 
Security Considerations HOW-TO 
Manager Application HOW-TO Naar hppa 
Clustering/Session Replication HOW-TO eamanage _ 
Developer Quick start 
Tomcat setup RealmsaAAA Seriet Exomples 
Firat Web Application JDBC DataSources 


Temcat Versions 


图 2-15 ” Tomcat 7.0 默认 的 首页 


2.3 任务 三 : Web 项 目的 创建 


MyEclipse 企业 级 工作 平台 是 对 Eclipse IDE 的 扩展 ,是 目前 最 流行 的 Java Web 集 
成 开发 环境 ,可 以 帮助 项 目 开 发 人 员 极 大 地 提高 工作 效率 。MyEclipse 的 安装 较 容易 , 按 
照 安 装 向 导 进 行 默认 安装 即 可 ,请 读者 自行 下 载 安装 


下 面 将 引导 读者 逐步 熟悉 JSP+Java 的 Web 项 目 在 MyEclipse 9.0 中 从 开发 到 发 布 
运行 的 操作 流程 。 


“任务 目标 : : 


在 MyEclipse 中 ,采用 JSP 十 Java 类 的 方式 完成 用 户 分 组 的 判断 。 
当 用 户 在 浏览 器 的 文本 框 中 输入 1 时 ,在 浏览 器 中 输出 “你 好 ,普通 用 户 1”; 
当 用 户 在 浏览 器 的 文本 框 中 输入 2 时 ,在 浏览 器 中 输出 “你 好 ,集团 用 户 1” 


: 当 用 户 在 浏览 器 的 文本 框 中 输入 3 时 ,在 浏览 器 中 输出 “你 好 ,VIP 用 户 1”。 
”技能 训练 : 


MyEclipse 的 使 用 。 
下 面 开 始 创建 一 个 Java Web 项 目 。 
第 一 步 ” 选 择 工作 空间 


第 一 次 进入 MyEclispe, 会 被 要 求 选择 工作 空间 (Workspace) , 即 项 目 源 代码 所 在 的 
物理 磁盘 上 的 一 个 目录 。 一 般 情况 下 ,不 要 把 工作 空间 建 在 系统 盘 上 。 下 面 的 例子 中 , 工 
空间 被 设置 为 E 盘 的 eae 目录 ,如 图 2-16 所 示 。 
第 二 步 ” 创 建 Web 项 


ee ite MyEclipse 开发 环境 (首次 进入 MyEclipse 会 显示 欢迎 


页 ,直接 将 它 关 闭 就 可 以 了 ) ,可 以 创建 新 的 Web 项 目 了 ,如 图 2-17 所 示 。 
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Select a workspace 
pt (]) 单 击 Browse 按 钮 ， 

在 弹出 的 资源 浏览 器 

ee: wos 选择 工作 空间 目录 


Dyse this ss the defoult and do not ask ssin (2) 单 击 OK 按 钮 


车 


图 2-16 选择 工作 空间 


各 


(3) 单 击 级 联 菜单 项 
Web Project 


的 


MyEclipse Java Enterprise 一 WyEclipse Enterprise Workbench 
(1) 单 击 File 菜 单 Edit Source Refactor Navigate Sesrch Project NyEclipse Run Window Help 
(2) 单 击 菜单 项 中 的 New 命 令 瑟 AttShiEt * TE 


Open File- Enterprise Application Project 


着 Yeb Service Project 


图 2-17 创建 Web 项 目 


在 接 下 来 出 现 的 对 话 框 中 填写 项 目 名 称 等 信息 ,如 图 2-18 所 示 。 


Create a Web Project 
Create a web project in the workspace or an external location 


-Web project Details 

Project Name: |test 一 一 一 一 一 一 (1) 在 Project Name 中 输入 项 目 名 称 

Location: 克 Use default location 
Directory: ~ [ENmyprojecttest | browse- 项 目 中 存放 源 代码 的 子 目录 ， 一 般 

Source folder |src 保持 默认 设置 src 

Web root folder: [webRoot Web 项 目 根 目录 ， 一 般 保持 默认 即 可 

ContextrootURL:[/est 一 上 下 文 路 径 ， 一 般 保持 默认 即 可 

FJ2EE Specification Level Java EE 版 本 规范 选择 ， 一 般 保持 默认 即 可 


G Java EE 5.0 J2EE14 J2EE13 


FMaven: 
厂 Add Maven support 
Learn more about Maven4MyEclipse.. 
FJSTLSupport 
GE Add JsTL libraries to WEB-INF/lib folder? 


-(2) 单 击 Finish 按 钮 


@ < Back Next > Cancel 


图 2-18 创建 Web 项 目 
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第 三 步 ” 新 建 包 

项 目 中 的 Java 类 文件 通常 要 用 “ 包 ” 来 进行 分 类 组 织 , 所 以 在 建立 了 一 个 新 项 目 后 ， 
要 为 这 个 项 目 新 建 包 ,如 图 2-19 所 示 。 


Fle Edit Source Refactor Navigate Search Project MyEclipse Run Window Help 


和 > 回避 全 ] 作 ”| 痢 古 |] 信 | 性 ”用 7|] 可 有 ”9 | 记 喇 |]@v| 国 
[2 


(1) 右 击 项 目 名 
(2) 单 击 菜单 中 的 New 


因 必 B Project 
其 Enterprise Application Project 
Open in New Window ee 


鞋 Web Service Project 
Open Type Hierarchy Fm 
Alt+shit+ Ww »| SJava Project 
国 Report Web Project 
5 [3project.. 
钨 Copy Qualified Name 
(3) 单 击 级 联 菜单 中 的 Package | 


图 2-19 创建 Web 项 目 


在 接 下 来 出 现 的 对 话 框 中 需要 填写 包 的 名 称 。Java 包 的 名 字 中 的 字母 通常 用 小 写 ， 
为 了 保障 每 个 Java 包 命名 的 唯一 性 ,要求 在 自 定义 的 包 的 名 称 之 前 加 上 唯一 的 前 绥 。 由 
于 互联 网 上 的 域名 称 是 不 会 重复 的 ,所 以 一 般 采 用 互联 网 上 的 域名 反 序 作为 自 定义 包 名 
的 唯一 前 级 ,如 图 2-20 所 示 。 


Java Package 


Create a new Java package. 


Creates folders corresponding to packages. 
项 目的 源 文件 目录 ， 一 般 保 持 默 认 olger [test/src 


(1) 设置 对 话 框 中 的 Name 来 为 包 命名 Namer com test www.packagel 


(2) 单 击 Finish 按 钮 


图 2-20 包 命名 
第 四 步 ” 新建 类 
包 被 创建 好 之 后 ,可 以 进一步 在 包 中 创建 Java 类 ,如 图 2-21 所 示 。 
BB test 
Bsrc 
(1) 右 击 包 名 一 一 一 一 一 一 | 忌 BBprojed 
fis ication 
(2) 单 击 菜单 项 中 的 New i aea 
由 ee wa OPA Mf in Web service Project 
Open Type Hierarchy 局 才 java Project 
Show in Alt+Shift+W 中 国 Repor web Project 
镜 copy cultc [I Project.. 
芥 copy Qualifed Name de | 
(3) 单 击 级 联 菜单 项 中 的 Class = 
图 2-21 新 建 类 
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在 接 下 来 出 现 的 New Java Class 对 话 框 中 填写 类 的 名 称 等 信息 ,如 图 2-22 所 示 。 注 
意 Java 类 的 命名 规范 ,应 该 以 大 写字 母 开 头 。 


(D 输入 类 的 名 称 


图 2-22 类 命名 


类 Customer 创建 完成 后 ,就 会 出 现在 包 中 。 下 面 按照 任务 一 的 要 求 来 编写 
Customer, java 文件 ,完成 客户 级 别 的 选择 ,如 示例 2-1 所 示 。 
闪 示 例 2-1 源 文件 : Customer. java 


Package com.test.www; 


Public class Customer 
{ 
public String getCustomerGroup (int group) 
{ 
String WelcomeMessage=" 你 好 ,访客 "; 
switch(group) 
{ 
case 1: WelcomeMessage=" 你 好 ,普通 用 户 "; 
break; 
case 2: WelcomeMessage=" 你 好 ,集团 用 户 "; 
break; 
case 3: WelcomeMessage=" 你 好 ,VIP 用 户 "; 
break; 
} 
return WelcomeMessage; 
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第 五 步 ”新建 JSP 文 件 
要 完成 任务 一 的 要 求 ,还 需要 编写 JSP 文件 ,以 便 在 浏览 器 中 进行 显示 。 下 面 新 建 
一 个 JSP 文件 ,创建 的 步骤 如 图 2-23 所 示 。 


日 葬 test 
白嫩 sr 
自 克 comtestwww.packagel 


(1) 右 击 项 目的 WebRoot 目 录 一 | 


(2) 单 击 菜单 项 中 的 New 


(3) 单 击 级 联 菜单 项 中 的 JSP 
图 2-23 ”新建 JSP 文件 


在 接 下 来 出 现 的 Create a new JSP page 对 话 框 中 输入 JSP 文件 的 名 称 , 如 图 2-24 
所 示 。 


JSP Wizard 


(1) 输入 JSP 文 件 的 名 字 


Template to use: | Default Jsp template 7| 


(2) 单 击 Finish 按 钮 2 one | 


图 2-24 命名 JSP 文 件 


创建 完 index. jsp 之 后 ,编写 代码 完成 用 户 输入 信息 的 获取 ,并 调用 示例 2-1 中 类 的 
方法 ,完成 用 户 组 别 的 判断 ,再 将 结果 显示 输出 。 如 示例 2-2 所 示 。 
闪 示 例 2-2” 源 文件 : index. jsp 


<%Q@page language="java" pageEncoding="GB18030" 
import="com.test.www.packagel.* ,java.util.*" %> 


引入 Java 类 所 在 的 包 
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<form name= "form" method= "post" action=""> 

请 输入 客户 类 别 1: 普通 用 户 。2: 集团 用 户 3: VIP 用 户 

<input type="text" name= "group" /> 
</form> 

< 
String groupS=request .getParameter ("group"); 
try 
{ 
int group=Integer.parseInt (groups); 


Customer customer=new Customer () ;一 一 一 [创建 Customer 类 的 实例 


String mess=customer.getCustomerGroup (group); 

out .Print (mess); 调用 实例 的 getCustomerGroup 方 法 
}catch () 
{ 
out .print ("请 输入 1~ 3 之 间 的 一 个 整数 ") 
} 
和 > 


2.4 任务 四 : 在 MyEclipse 中 配置 Web 服务 器 


到 此 为 止 ,一 个 Java Web 项 目 已 经 建立 完成 了 。 但 是 如 果 要 运行 ,还 需要 Web 服务 
器 的 支持 。 本 案例 选用 Tomcat 7. 0 作为 Web 服务 器 。Tomcat 是 一 个 应 用 较 广 泛 且 为 
开源 的 Web 服务 器 ,其 安装 过 程 比较 简单 ,在 此 不 做 详细 介绍 ,请 读者 自行 下 载 并 安装 。 

服务 器 安装 好 之 后 ,可 以 在 操作 系统 中 启动 并 完成 相应 的 功能 。MyEclipse 中 也 集 
成 了 服务 器 的 设置 和 启动 等 功能 ,使 项 目的 开发 和 调试 更 加 方便 ,快捷 。 

任务 目标 ， 
为 MyEclipse 配置 Tomcat 7. 0 服务 器 ,配置 完成 后 ,在 MyEclispe 中 启动 服务 
| 器 ， 使 之 可 以 正常 运行 。 

: ”技能 训练 : 
: MyEclipse 的 使 用 。 

下 面 以 图 示 的 方式 ,逐步 介绍 如 何在 MyEclipse 中 配置 和 启动 Web 服务 器 。 

第 一 步 ” 配 置 Tomcat 7.0 服务 器 

在 MyEclipse 菜单 下 方 的 功能 按钮 中 找到 Run/Stop/Restart MyEclipse Servers 按钮 , 单 
击 右 侧 的 向 下 箭头 ,然后 再 单 击 菜单 项 中 的 Configure Server, 如 图 2-25 和 图 2-26 所 示 。 

在 图 2-26 所 示 菜 单项 中 单 击 Configure Server, 会 弹出 一 个 对 话 框 ,在 此 对 话 框 中 配 
置 Tomcat 7.0 服务 器 的 步骤 如 图 2-27 所 示 。 

第 二 步 ” 启 动 Tomcat 7.0 服务 器 

完成 第 一 步 的 配置 后 ,Tomcat 7. x 会 出 现在 MyEclipse 可 用 服务 器 列表 中 ,可 以 方 
便 地 启动 服务 器 ,如 图 2-28 所 示 。 
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单 击 向 下 箭头 


Fle Edit Navigate Search Project MyEclipse Run Window Help 


[BA LL EE A: 让 地 | 国生 
| 妇 7 了 四 7 了 7 守 YO [Run/Stop/Restart MyEclipse Servers| 


去 


图 2-25 ”MyEclipse 功能 按钮 


昌 ”@ | 名 二 |] 革 外” 


hh 的 Configure Server 


图 2-26 MyEclipse 服务 器 按钮 的 下 拉 菜 单项 


(5) 选中 Tomcat server 项 中 


的 Enable 单 选 按钮 
Mecipse 
(1) 单 击 左 仙 栏 poor Poem 
中 的 Servers 
oProoram Fles\Tomeat 70 Browse_- (4) 单 击 Tomcat home 
[Daprogram Fies\Tomcat70 EGR directory 的 
[Grprooram Fies\omeat 7 iemp | Bowe Browse.… 按 钮 
= 
Springsource 
由 Sunjav System £ 
(2) 单 击 菜 单 Sun Java System \ 
项 Tomcat 
(3) 单 击 级 联 菜单 中 
的 Tomcat 7.x 
(6) 单 击 OK 按钮 


图 2-27 服务 器 Preferences(Filtered) 对 话 框 


(DD 单 击 Servers 选 项 卡 


天 MyEdipse Derby 可 stopped 
威 MyEclipse Tomcat 本 stopped 


(2) 右 击 Tomcat 7.x 服 务 (3) 单 击 菜单 项 中 的 Run Server 
图 2-28 在 MyEclipse 中 启动 服务 器 
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启动 服务 器 后 ,在 Console 选项 卡 中 会 出 现 服务 器 启动 的 相关 信息 ,如 果 在 此 过 程 中 
出 错 ,Console 中 会 出 现 错误 提示 ;如果 正常 ,服务 器 状态 将 由 Stopped 变 为 Running。 


2.5 任务 五 : 向 Web 服务 器 上 部 署 项 目 


服务 器 正常 启动 后 ,在 浏览 器 的 网 页 地 址 栏 中 输入 http://localhost:8080, 就 可 以 看 
到 Tomcat 7. 0 的 默认 主页 了 。 但 是 现在 还 不 能 够 访问 本 章 任务 一 编写 的 JSP 文件 ,要 
使 任务 一 中 完成 的 项 目 运行 起 来 ,还 需要 将 项 目 部 署 到 服务 器 上 。 下 面 以 图 示 的 方式 引 
导读 者 完成 项 目的 部 署 。 

第 一 步 ” 设 置 项 目 发 布 的 目标 目录 

项 目 部 署 到 服务 器 上 的 目标 目录 可 以 由 开发 人 员 通 过 设置 项 目 属性 来 指定 ,如 图 2-29 
和 图 2-30 所 示 。 


(1) 右 击 项 目 名 ET 
日 名 
[Ee 
由 融 | Openin New Window 

日 多 | Open Type Hierarchy ma 

Show In Alt+Shift+W » 


es be » 
Image| Configure » 
(2) 单 击 菜单 项 中 的 Properties 一 [5 于 E7770 


图 2-29 设置 项 目 属性 


第 二 步 ” 设 置 项 目 部 署 的 目标 目录 

如 图 2-31 和 图 3-32 所 示 是 向 服务 器 上 部 署 项 目的 向 导 图 示 。 

单 击 Add Deployment 命令 后 会 弹出 一 个 对 话 框 ,如 图 2-32 所 示 。 

项 目 部 署 成 功 后 ,会 出 现在 服务 器 下 方 , 状 态 为 OK, 如 图 2-33 所 示 。 

本 章 的 三 个 任务 都 完成 后 ,就 可 以 在 下 浏览 器 中 访问 项 目的 index. jsp, 从 而 完成 用 
户 组 的 判断 了 。 
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两 百 梧 
3 [| Tor 
G) 输入 目标 目录 “7” | we -一 
代表 部 署 到 服务 器 一 | es sa 
的 根 目录 下 me Code sys oer [reboot E39 
Re ‘Web Context-root (see Notice) 
Javadoc Location 
(D) 单 击 左 侧 菜 单项 中 一 上 -> we = 
的 MyEclipse ep orient root changes only pp 1o exploded and padaged WAR deployments of this project 
Task Tags To update the context.root when this project is deployed as a module of an enterprise project 
uw FRR 的 
(2) 单 击 二 级 菜单 项 一 一 人 Te 光合 全 oa po 
中 的 Web -有 speciling the new comtextroot 
praject References 
Run/Debug Settings 
(4) 单 击 OK 按钮 
| 
图 2-30 设置 项 目 属性 
(1) 右 击 服务 器 
车 Configure Server Connector 
萝 Debug Server 
© Run server 
% Restart Server 
国 Stop Server 
(2) 单 击 菜单 项 中 的 Add 
Deployment... 
Manage Launch Configurations... 
Run Configurations... 
Debug Configurations... 
图 2-31 设置 项 目 属性 


New Deployment 
Create new project deployment for Tomcat 7x 


EE 


Server: Tomcat 7x 
(1) 在 Project 列 表 菜 单 中 | Eee be 
选择 要 部 署 的 项 目 
Deploytype: & © Exploded Archive (development mode) © Packaged Archive (production mode) 
Deploy Location: [DAProgram FlesTomcat70WebappsROOT 
(2) 单 击 Finish 按 钮 


@ 七 mn ] come | 


图 2-32 ”选择 要 部 署 的 项 目 
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Fi WOK BExploded _ DAprogram Files\Tomcat 7.0\webapps\ROOT 


图 2-33 项 目 部 署 成 功 


课 后 练习 


1. 请 读者 下 载 Tomcat 6.0 和 MyEclipse 6.0 以 上 版 本 并 完成 安装 。 

2. 采用 JSP 十 Java 类 的 方式 完成 累加 计算 : 根据 用 户 输入 的 数字 n, 在 浏览 器 中 显 
示 1~n 累加 的 值 。 

要 求 : 在 MyEclipse 中 创建 一 个 Java Web 项 目 , 实 现 题目 的 要 求 , 并 成 功 部 署 到 服 
务 器 上 。 

3. 采用 JSP 十 Java 类 的 方式 完成 对 用 户 输入 的 两 个 浮 点 数 的 四 则 运算 : 用 户 输入 两 
个 浮 点 数 ,并 选择 四 则 运算 中 的 一 种 ,在 浏览 器 中 显示 运算 结果 。 

要 求 : 在 MyEclipse 中 创建 一 个 Java Web 项 目 , 实 现 题目 的 要 求 ,并 成 功 部 署 到 服 
务 器 上 。 

提示 : 字符 串 转 浮 点 的 方法 : float {f 王 Float. parseFloat(String s)。 

4. 在 MyEclipse 中 分 别 为 “ 佳 衣 屋 ” 和 第 1 章 中 的 “自选 项 目 " 创 建 一 个 Web 
Project ,为 后 续 章 节 做 好 准备 工作 。 
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本 章 学 习 要 点 : 

。 熟练 掌握 在 SQLyog 9.0 中 设计 与 实现 MySQL 数据 库 的 方法 ; 
。 熟练 掌握 数据 库 导 入 与 备份 的 方法 ; 

。 熟练 掌握 数据 库 的 连接 池 技 术 。 


数据 库 的 设计 与 开发 是 Web 项 目 开 发 不 可 缺少 的 一 个 环节 , Web 应 用 中 的 数据 都 
存储 在 数据 库 中 。 本 书 中 的 项 目 开 发 采用 了 MySQL 这 一 开源 数据 库 , 版 本 为 5. 1, 同 时 
采用 SQLyog 9. 0 作为 MySQL 数据 库 的 开发 管理 工具 ,请 读者 自行 下 载 并 安装 MySQL 
和 SQLyog 9.0。 本 章 介绍 如 何 使 用 SQLyog 9.0 来 开发 项 目的 数据 库 。 


3.1 任务 一 : 创建 数据 库 


”使 用 SQLyog 为 " 佳 衣 屋 "项 目 创建 MySQL 数据 库 。 
技能 训练， 
| SQLyog 的 用 法 。 


MySQL 和 SQLyog 都 安装 完毕 后 ,首先 打开 SQLyog 与 MySQL 服务 器 建立 连接 ， 
如 图 3-1 所 示 。 


EE Ea. 有 E52 LL 
ee | 


we Jer lm | mil] 
CE 

(1) 输入 安装 MySQL 时 设置 的 用 户 名 一 = 

(2) 输入 安装 MySQL 时 设置 的 密码 一 

(3) 输入 安装 MySQL 时 设置 的 端口 号 7] 


UW | 


wons wm 
MySQL: a my) 


(4) 单 击 “ 连 接 " 按 钮 - 


图 3-1 用 SQLyog 连接 MySQL 数据 库 
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连接 上 MySQL 服务 器 后 ,就 可 以 创建 数据 库 , 如 图 3-2 和 图 3-3 所 示 。 


多 文件 编 吉 收 襄 夫 数据 库 表单 其 他 本 具 高 级 T 具 窗口 孝 
TE EE 
ca x| 国 


(1) 右 击 左 栏 中 的 root@localhost ct FE 
日 目 informatid 4 局 条 对象 刘 览 各 
(2) 单 击 菜单 项 “创建 数据 库 ” EEEEE Ctrl+D 
加 到 和 演 索 Ctrl+Shift+D 


本 计划 备份 Ctrl+Al+S 
忆 号 入 外 部 数据 … Ctr+Ak+O 
Ctrl+Shift+Q 


(4) 单 击 “ 创 建 "按钮 


图 3-3 新 数据 库 的 基本 配置 


单 击 图 3-2 中 的 “创建 数据 库 ”" 菜 单项 后 ,SQLyog 弹出 一 个 对 话 框 ,要 求 对 新 数据 库 
做 一 个 简单 的 配置 ,如 图 3-3 所 示 。 
单 击 图 3-3 中 的 “创建 ?按钮 后 ,数据库 jyw 就 创建 好 了 。 


3.2 任务 二 : 创建 数据 表 


” 使 用 SQLyog 为 任务 一 中 创建 的 数据 库 jyw 创建 存储 商品 信息 的 数据 表 goods。 ， 
”技能 训练 : : 
” SQLyog 的 用 法 。 
创建 新 表 ( 见 图 3-4) 需 要 为 表 命名 ,并 进行 创建 列 ( 即 字段 ) 等 的 设计 ,如 图 3-5 所 示 。 
按照 图 中 的 步 又 操作 完成 后 ,数据 表 goods 就 创建 完成 了 。 本 书后 续 章 节 任 务 示例 
中 需要 的 数据 表 的 设计 如 表 3-1~ 表 3-5 所 示 , 请 读者 自行 完成 这 些 数据 表 的 创建 。 
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中 小 型 Web 项 目 开 发 实战 


(2) 选择 菜单 项 “ 创 / 建 
(D) 右 击 左 栏 中 的 jyw 数 据 库 (3) a 


图 3-4 创建 表 
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(0D 输入 表 的 名 称 (2) 选择 表 的 字符 集 
双 新 表 x| 国 
EE 绊 [us | 器 
%g 放 pr 加 ?Ht | 司 
yy [em 本 
加 1 列 
oe 
lia lvarcnar [El20 | 回 | 口 商品 编号 
oo lvarcnar [>]20 湛 国 Er 
limage lvarchar [ll20 | 商品 图 片 文件 名 
price [floae [= 商品 价格 
color lvarcnar 回 20 | 商品 闫 色 
[warchar [4 | | 和 证 [ee = 
保存 和 还原 
(3) 根据 第 1 章 中 的 需求 分 析 ， 设 计 表 goods 的 列 ， (4) 单 击 “ 保 存 "按钮 
注意 ; varchar 类 型 的 列 一 定 要 设置 长 度 ， 否 则 
表 不 能 被 成 功 创建 
图 3-5 数据 表 goods 的 设计 
表 3-1 员工 信息 数据 表 employee 
列 名 称 数据 类 型 说 明 
id int 序号 ,主键 .自动 增长 
account varchar 用 户 名 
password varchar 密码 
storenum int 所 属 分 店 
name varchar 真实 姓名 
authority int 权限 
表 3-2 销售 信息 数据 表 sales 
列 名 称 数据 类 型 说 明 
id int 序号 .主键 自动 增长 
goodsid varchar 商品 编号 
discount int 折扣 
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续 表 

列 名 称 数据 类 型 说 明 
date timestamp 销售 时 间 
quantity int 销售 数量 
salesperson varchar 销售 人 员 
size varchar 尺码 
storenum int 分 店 编号 

列 名 称 数据 类 型 说 明 
id int 序号 .主键 自动 增长 
stock int 库存 量 
size varchar 尺码 
goodsid varchar 商品 编号 
storenum int 分 店 编号 

表 3-4 门店 信息 数据 表 storeinfo 

列 名 称 数据 类 型 说 明 
id int 序号 .主键 自动 增长 
name varchar 分 店名 称 
addr varchar 分 店 地 址 
contact varchar 分 店 联系 方式 

列 名 称 数据 类 型 说 明 
id int 序号 .主键 自动 增长 
goodsid varchar 商品 编号 
quantity int 进货 数量 
storenum int 分 店 编号 
applydate timestamp 请 求 时 间 
processdate timestamp 处 理 时 间 
size varchar 尺码 
orderstatus varchar 订单 状态 : 待 处 理 / 已 发 货 / 待 发 货 /取消 
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“任务 目标 
: 为 新 创建 的 数据 表 goods 添加 新 记录 。 
”技能 训练 : 


。 SQLyog 的 使 用 ; 

。 数据 表 中 记录 的 添加 ; 
。 数据 表 中 记录 的 删除 ; 
。 保存 数据 表 的 修改 。 


下 面 为 数据 表 goods 添加 新 记录 ,如 图 3-6 所 示 。 
(1) 右 击 goods 数 据 表 


(2) 单 击 菜单 项 "打开 表 "命令 
图 3-6 打开 数据 表 


数据 表 在 右 侧 栏 中 被 打开 ,在 各 列 中 输入 相应 的 信息 就 可 以 添加 新 记录 了 ,如 图 3-7 所 示 。 


万 限 制 和 第 - 行 : .J | 行 


图 3-7 录入 表 中 的 记录 


如 果 要 删除 一 条 记录 ,选中 该 记录 前 面 的 复 选 框 , 单 击 红色 垃圾 箱 标 识 的 Delete 
selected row(s) 功 能 键 即 可 ,如 图 3-8 所 示 。 


(2) 单 击 Delete selected row(s) 功 能 键 


YH12CEN1 .jpg 


YR12CKY2. pg 


(0D 选中 要 删除 的 记录 前 的 复 选 框 


图 3-8 删除 表 中 的 记录 
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数据 表 中 的 数据 改变 后 ,应 及 时 保存 : 单 击 软盘 标识 的 Save changes 功能 键 即 可 ,如 
图 3-9 所 示 。 


单 击 Save changes 功 能 键 


历 限制 行 党 - 行 : :J | 和 


和 牛仔裤 JYR12CKY2.jpg | 158| 黄 S-XL 


图 3-9 保存 数据 表 所 做 的 修改 


3.4 任务 四 : 创建 视图 


任务 二 中 创建 的 基本 信息 表 并 不 能 满足 所 有 的 应 用 需求 ,有 时 ,所 需要 的 信息 是 从 两 
个 或 多 个 基本 表 中 得 来 的 ,这 就 需要 用 视图 来 表示 。 例 如 要 得 到 销售 商品 的 信息 ,其 中 一 
部 分 信息 (如 销售 时 间 ) 从 销售 情况 表 中 得 来 , 另 一 部 分 从 商品 信息 表 中 得 来 的 (如 商品 名 
称 ) 。 本 节 讲 授 在 SQLyog 中 如 何 创建 视图 。 

”任务 目标 : | 
”创建 视图 salesinfo, 其 中 商品 编号 名称、 图片 颜色、 价格 来 自 数据 表 goods, 销 ， 
; 售 日 期 尺码、 数量 .销售 人 员 、 销 售 数量 来 自 数据 表 sales,goods 表 中 的 指导 价格 乘 以 : 
”sales 表 中 的 折扣 再 除 以 100, 即 为 视图 中 的 销售 价格 。 
”技能 训练 : 
: 。 SQLyog 的 使 用 ; 

。 视图 的 创建 ; 

。 SQL 脚本 的 编写 。 

图 3-10 一 图 3-12 以 图 形 向 导 的 方式 介绍 了 如 何在 SQLyog 中 创建 一 个 视图 ,由 于 纺 
写 SQL 命令 集 ( 脚 本 ) 属 于 数据 库 课程 的 内 容 ,在 本 书 中 不 做 详细 讲解 。 


(D) 右 击 “ 视 图 "菜单 项 
[62) 第 市 级 家 区 尝 现 "名 寻 视图” 


E 


i 


国 国 田 田 田 田 


BE 


图 3-10 创建 视图 图 3-11 输入 新 视图 名 称 
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O) We (1) 编写 SQL 脚本 
ET EL EE 


We root@localhost 加 =alesinfo x | 国 
占 information schem ; [Tab]-> 下 一 个 村 区，[CySpscej> 列 下 所 有 标 要 [CEWter> 列 卫 区 本 标本 
日 目 jyw 


DELIMITER $$ 
UsE “jyw'$s 
DROP VIEN IF EXISTS “salesinfos$ 


CREATE ALGORITIN-UNDEF INED DEFINER®"root"g"localhost™ SQL SECURITY DEFINER VIPN “saleatnfo” AS ( 
SELECT 


生计 


加 
加 
加 
图 
日 
日 
a 


因 


ERESeecsaowewnn 


ice*) / 100) AS “aellingprice 


WHERE ("goods. "id" = “sales*."goodsid"))$5 


DELIMITER ; 


图 3-12 编写 SQL 脚本 并 生成 视图 


3.5 任务 五 : 数据 表 的 备份 和 还 原 


数据 库 要 经 常 性 地 进行 备份 ,在 万 一 出 现 数据 丢失 的 情况 下 还 可 以 根据 备份 文件 进 
行 数据 的 恢复 。 


: 使 用 SQLyog 为 jyw 数据 库 进行 备份 ,并 可 由 备份 进行 数据 的 还 原 。 
; 技能 训练 : 
: 。 SQLyog 的 使 用 方法 ; 
。 数据 库 的 备份 和 还 原 。 
如 图 3-13 和 图 3-14 所 示 是 SQLyog 中 进行 数据 库 的 操作 步骤 。 


(1) 右 击 项 目 名 (2) 单 击 菜单 项 中 的 “备份 /导出 "命令 (3) 单 击 级 联 菜单 项 中 的 “备份 数 
据 库 ， 人 


图 3-13 数据 库 的 备份 
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SQLS 出 个 结构 唯一 个 仅 有 教 据 。 人 结构 和 数据 


孝 据 库 名 称 导出 到 文件 
:wo 一 (1) 输入 或 者 选择 导出 的 目标 SQL 文件 


(2) 单 击 “ 导 出 "按钮 


图 3-14 导出 SQL 文件 
完成 以 上 的 步骤 后 ,数据 库 的 备份 就 完成 了 。 
可 以 把 数据 库 jyw 删除 ,再 通过 图 3-15 一 图 3-17 的 操作 进行 恢复 。 
(1) 右 击 用 户 root@localhost 


(2) 单 击 菜单 项 "执行 SQL 
脚本 .…” 


图 3-15 编写 SQL 命令 生成 视图 


单 击 图 3-15 中 的 菜单 项 “执行 SQL 脚本 .….” 后 ,出现 如 图 3-16 所 示 的 对 话 框 。 


(D 输入 或 者 选择 数据 库 
的 备份 SQL 文件 


(2) 单 击 "执行 "按钮 


图 3-16 ”编写 SQL 命令 以 便 生成 视图 


单 击 图 3-16 中 的 “执行 ”按钮 后 ,会 出 现 一 个 进度 条 ,提示 数据 恢复 的 进度 。 恢 复 完成 
后 ,还 需要 进行 主动 刷新 ,恢复 的 数据 库 才 会 出 现在 左 侧 的 对 象 浏览 器 中 ,如 图 3-17 所 示 。 
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(1) 右 击 用 户 root@localhost 
(2) 单 击 菜单 项 “刷新 对 象 


浏览 器 ” 


图 3-17 刷新 对 象 浏览 器 


在 刷新 对 象 浏览 器 中 可 以 看 到 恢复 的 数据 库 , 读 者 可 以 打开 数据 库 的 数据 表 ,查看 数 
据 是 否 与 做 备份 时 相同 。 


3.6 任务 六 : Tomcat 数据 库 连 接 池 的 配置 


数据 库 连 接 是 一 种 关键 且 有 限 的 资源 ,这 一 点 在 多 用 户 的 Web 应 用 中 体现 得 尤为 突 
出 。 对 数据 库 连接 的 管理 能 显著 影响 到 整个 应 用 程序 的 可 扩展 性 和 健壮 性 ,影响 到 Web 
应 用 的 性 能 指标 。 数 据 库 连接 池 正 是 针对 这 个 问题 提出 来 的 。 数 据 库 连接 池 负 责 分 配 、 
管理 和 释放 数据 库 连 接 , 它 允许 应 用 程序 重复 使 用 一 个 现 有 的 数据 库 连 接 , 而 不 是 重新 建 
立 一 个 ;通过 释放 空闲 时 间 超 过 最 大 空闲 时 间 的 数据 库 连 接 来 避免 因为 没有 释放 数据 库 
连接 而 引起 数据 库 的 连接 问题 。 这 项 技术 能 明显 提高 数据 库 操作 的 性 能 。 

为 Tomcat 配置 MySQL 数据 库 连 接 池 .需要 在 Tomcat 的 context. xml 文件 中 的 
二 context 之 过 /context 志 之 间 添 加 MySQL 连接 池 配 置 ,操作 如 下 。 

<Resource name="jdbc/mysql" 

auth="Container" 


type="javax.sql .DataSource" 


maxActive= DR 最 大 活跃 连接 ， 设 置 为 0 时 表示 无 限制 


_， ， “了 没有 数据 库 连 接 时 ， 连 接 池 也 必须 保持 最 大 空闲 连接 数 ， 
maxIdle- "10" 一 一 设置 为 /表示 无 限制 


maxWait="5000" 连接 建立 等 待 时 间 ， 单 位 为 毫秒 ， 设 置 为 -1 表示 无 限制 


= 出 使 用 数据 库 的 用 户 名 
username= "root 
Password= "passwordn 使 用 数据 库 的 用 户 密码 


driverClassName="org.gjt.mm.mysql .Driver" 


url="jdbc:mysql://localhost/jyw" /> 连接 数据 库 的 URL 


与 MySQL 的 连接 还 需要 MySQL 驱动 的 支持 ,请 读者 自行 下 载 MySQL 的 jdbc 驱 
动 jar 包 mysql-connector-java-5. 0. 2-beta-bin.jar, 并 将 其 复制 到 tomcat 下 的 lib 目录 下 。 

完成 本 章 任务 六 后 ,数据 库 的 连接 池 配 置 就 完成 了 。 需 要 注意 的 是 ,任务 六 中 的 配置 
是 针对 Tomcat 7.0,Tomcat 6.0 及 以 下 版 本 的 配置 方法 会 有 不 同 。 作 为 开源 Web 服务 
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器 ,Tomcat 得 到 了 广泛 的 应 用 ,针对 Tomcat 的 技术 讨论 在 互联 网 上 也 很 活跃 ,请 使 用 
Tomcat 6.0 及 以 下 版 本 的 读者 自行 查找 资料 并 完成 连接 池 的 配置 。 


课 后 练 习 


1. 请 读者 下 载 MySQL 5. 0 及 以 上 版 本 、SQLyog 9. 0 及 以 上 版 本 并 完成 安装 。 

2. 请 按照 本 章 任务 二 中 的 表 3-1 一 表 3-5 完成 各 数据 表 的 创建 ,并 进行 信息 的 初始 
化 录入 。 

3. 在 练习 2 的 基础 上 ,从 基本 表 goods 和 orders 中 导出 订单 详细 信息 视图 
ordersinfo 。 

要 求 : 商品 的 编号 、 名 称 图片 .颜色 .价格 来 自 商 品 信息 数据 表 goods, 序 号 .尺码 、 数 
量 、 请 求 日 期 ,处理 日 期 .订单 状态 、 分 店 号 来 自 订单 数据 表 orders, 其 条 件 为 goods 表 中 
的 id 与 orders 表 中 的 goodsid 相等 。 

提示 : 参考 本 章 任务 四 ,完成 视图 ordersinfo 的 创建 。 

4. 在 练习 2 的 基础 上 ,导出 详细 库存 信息 视图 stockinfo。 

要 求 : 以 商品 信息 表 goods 库存 信息 表 stock 分 店 信息 表 storeinfo 为 基本 表 , 导 出 
图 3-18 中 的 视图 。 


YW12CRN1 | 牛仔 涛 |JYW12CKN1.jpg| 123| 水 尝 因 四 便衣 层 总 店 华强 北 店 38| 

ywlzcFY2| 和 牛仔 柱 |TYW12CFY2.jpg| 。 158| 黄 佳 衣 屋 总 店 化 强 北 店 30| 

JYW12CRKN1 | 牛仔 注 |JYW12CKN1. jpg 123| 水 尝 畸 攻 佳 农 屋 海 岩 城 店 20 

(NULL) | (NULL) |(NULL) (NULL) | (NOLL) |(NULL) |(NULL) (NULL) 
图 3-18 视图 


提示 : 参考 本 章 任 务 四 ,完成 视图 stockinfo 的 创建 。 
5. 根据 第 1 章 中 的 需求 分 析 , 完 成 “ 佳 衣 屋 " 项 目 数据 库 的 设计 。 
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本 章 学 习 要 点 : 

。 熟练 掌握 表单 及 其 常用 元 素 的 用 法 ; 

。 熟练 掌握 JSP 指令 标签 的 用 法 ; 

。 熟练 掌握 JSP 动作 标签 的 用 法 ; 

。 熟练 掌握 内 置 对 象 request 的 常用 方法 的 用 法 ; 
。 熟练 掌握 内 置 对 象 response 的 常用 方法 的 用 法 ; 
。 熟练 掌握 内 置 对 象 session 的 常用 方法 的 用 法 ; 
。 熟练 掌握 内 置 对 象 application 的 常用 方法 的 用 法 ; 
熟练 掌握 内 置 对 象 out 的 常用 方法 的 用 法 ; 

。 了解 JDBC 的 结构 和 原理 ; 

熟练 掌握 JDBC 操作 数据 库 的 步骤 。 


本 章 内 容 涉及 HTML 表单 及 其 元 素 、JSP 基本 概念 .JSP 内 置 对 象 和 JDBC 数据 库 
操作 等 ,是 后 面 章节 中 的 “ 佳 衣 屋 ?项目 开发 过 程 中 所 使 用 的 技术 的 知识 基础 ,已 经 熟悉 本 
章 内 容 的 读者 可 以 直接 进入 第 5 章 的 学 习 。 或 者 在 后 续 章 节 的 项 目 开 发 中 遇 到 不 明白 的 
概念 时 ,再 回 到 本 章 寻 找 参 考 内 容 。 


4.1 HTML 表单 及 其 元 素 


HTML 表单 和 元 素 用 于 为 Web 应 用 搜集 不 同类 型 的 用 户 输入 ,表单 元 素 必须 包含 
在 表单 中 才能 有 效 使 用 。 


4.1.1 表单 


表单 是 一 个 包含 表单 元 素 的 区 域 , 表 单 使 用 表单 标签 (二 form 之 一 /form>) 定 义 。 
表单 可 选 的 属性 见 表 4-1 。 

(1) action 属性 。 用 于 指定 此 表单 数据 提交 的 目标 URL. 它 通常 是 一 个 相对 路 径 。 
如 果 没 有 设置 这 个 属性 或 者 属性 值 为 空 ,表单 数据 将 提交 给 此 表单 自身 的 URL。 


表 4-1 表单 属性 
属 性 值 描 述 
action URL 确定 当 提 交 表单 时 ,向 何 处 发 送 表单 数据 
accept MIME_type 确定 通过 文件 上 传 来 提交 的 文件 的 类 型 
accept-charset charset 服务 器 处 理 表 单数 据 所 接受 的 字符 集 
enctype MIME_type 确定 表单 数据 在 发 送 到 服务 器 之 前 应 该 如 何 编码 
method get| post 确定 如 何 发 送 表 单数 据 
name name 确定 表单 的 名 称 
target -blank| =parentj .self| 确定 在 何 处 打开 action URL 
_top|framename 


(2) method 属性 用 于 指定 提交 表单 数据 的 方式 ,常用 的 有 GET 和 POST 两 种 方式 。 
如 果 没 有 设置 此 属性 或 者 此 属性 值 为 空 , 则 使 用 GET 方式 来 提交 数据 。GET 和 POST 
提交 方式 的 主要 不 同 点 在 于 如 下 三 点 。 

@ 因为 GET 数据 是 URL 的 一 部 分 ,所 以 它 会 将 表单 数据 附 在 URL 后 面 传送 。 也 
就 是 说 ,在 浏览 器 的 地 址 栏 将 会 显示 表单 中 的 数据 ,并 且 在 通常 情况 下 ,浏览 器 会 将 这 个 
附加 数据 后 的 URL 保存 起 来 ,可 以 通过 浏览 器 的 “历史 ”来 得 到 它 。 所 以 ,这 种 方式 不 适 
合 于 发 送 需 要 保密 的 数据 的 表单 ,比如 密码 等 。 而 POST 不 是 URL 的 一 部 分 ,所 以 它 不 
会 将 表单 数据 附 在 URL 后 面 ,所 以 这 种 方式 不 会 发 生 上 面 的 问题 。 从 这 种 意义 上 讲 ， 
POST 方式 比 GET 方式 更 安全 。 

@ 因为 浏览 器 通常 会 限制 URL 的 长 度 , 所 以 使 用 GET 这 种 方式 无 法 传送 大 量 的 数 
据 。 而 POST 方式 不 会 有 这 种 问题 ,所 以 POST 方式 传输 的 数据 量 大 。 

@ 由 于 GET 传输 的 数据 量 小 ,所 以 有 些 数据 类 型 ,比如 文件 的 传输 只 能 用 POST 方 
式 。 从 这 一 点 上 讲 , POST 方式 可 传输 的 数据 类 型 更 多 。 

所 以 ,如 果 没 有 特殊 的 需要 ,应 使 用 POST 方式 来 传送 表单 数据 。 

(3) name 属性 。 用 于 给 这 个 FORM 指定 一 个 名 字 , 可 以 用 字母 和 数字 组 合 的 方式 
来 给 FORM 命名 ,但 不 要 用 数字 开头 。 

(4) enctype 属性 。 用 于 定义 数据 在 发 送 前 需要 完成 的 编码 方式 ,如 果 没 有 设置 这 个 
属性 ,那么 会 使 用 默认 的 值 application/x-www-form-urlencoded, 它 使 用 的 编码 方式 是 
UTF-8。 

(5) accept 属性 。 用 于 指定 处 理 表单 数据 的 ASP、JSP、Servlet 或 者 其 他 程序 接受 的 
MIME(Multipurpose Internet Mail Extension protocol, 多 用 途 网 际 邮件 扩充 协议 ) 数 据 
类 型 ,如 果 FORM 中 有 文件 组 件 (FILE) ,还 可 以 使 用 它 来 限制 上 载 文 件 的 类 型 。 

(6) accept-charset 属性 。 用 于 指定 处 理 表 单数 据 的 ASP、JSP、Servlet 或 者 其 他 的 
程序 接受 的 字符 编码 。 


4.1.2 表单 元 素 


表单 元 素 在 Web 应 用 中 用 来 给 访问 者 填写 信息 ,从 而 能 采集 客户 端 信息 ,使 Web 应 
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用 具有 交互 的 功能 。 常 用 的 表单 元 素 有 以 下 几 种 。 
1. 文本 框 
文本 框 定义 的 基本 格式 如 下 : 


<INPUT TYPE="TEXT" NAME="" VALUE="" SIZE="" MAXLENGTH=""> 


文本 框 用 标记 INPUT 定义 ,并 且 需 要 将 它 的 TYPE 属性 值 定义 为 TEXT, 这 也 是 
INPUT 标记 的 默认 类 型 ;NAME 属性 用 于 给 文本 框 指定 一 个 名 字 , 这 个 属性 是 必需 的 ; 
VALUE 属性 可 以 用 于 指定 文本 框 的 默认 值 ;SIZE 属性 用 于 定义 文本 框 的 大 小 ,默认 是 
20; 而 MAXLENGTH 是 用 于 限制 文本 框 的 输入 数据 长 度 的 属性 。 


2. 密码 框 


密码 框 和 文本 框 类 似 , 和 文本 框 唯一 的 区 别 是 ,需要 将 INPUT 标记 的 TYPE 属性 设 
置 为 PASSWORD, 其 他 的 属性 设置 和 文本 框 的 设置 一 样 。 另 外 ,在 密码 框 中 输入 数据 的 
时 候 , 密 码 框 中 不 会 明文 显示 输入 的 数据 ,而 是 用 * x ”或 其 他 的 掩盖 字符 来 表示 ,但 这 并 
不 影响 将 输入 的 数据 进行 发 送 。 


3. 文本 域 
文本 域 使 用 二 TEXTAREA 二 标记 来 定义 , 它 的 基本 格式 如 下 : 


<TEXTAREA NAME="" ROWS="" COLS=""></TEXTAREA> 


与 前 面 的 两 个 文本 元 素 不 同 , 二 TEXTAREA 放 标记 必须 成 对 出 现 , 它 有 三 个 属性 必 
须 定 义 : NAME 用 于 设置 文本 域 的 名 字 , ROWS 用 于 设置 文本 域 的 行 数 ,而 COLS 用 于 
设置 文本 域 的 列 数 , 如 果 文 本 域 有 默认 值 , 则 将 默认 值 放 在 二 TEXTAREA 之 和 
< /TEXTAREA> 之 间 。 


4. 列表 菜单 


列表 菜单 向 用 户 提 供 一 系列 的 选项 。 它 可 以 分 为 单 选 列表 和 多 选 列表 两 种 。 单 选 列表 
可 以 让 用 户 选 择 一 个 选项 , 它 也 是 下 拉 列 表 的 默认 设置 ;多 选 列表 可 以 让 用 户 选 择 多 个 

列表 菜单 的 基本 格式 如 下 : 

<SELECT NAME="" SIZE=""> 

<OPTION VALUE= "#">#< /OPTION> 

</SELECT> 

列表 菜单 使 用 SELECT 标记 来 定义 ,需要 使 用 NAME 属性 来 给 它 指定 一 个 名 字 。 
SIZE 属性 用 于 指定 下 拉 列 表 在 浏览 器 中 显示 的 行 数 ,如 果 不 指 定 这 个 属性 ,那么 在 浏览 
器 中 只 有 一 行 可 见 。 下 拉 列 表 的 各 个 选项 可 以 使 用 和 OPTION 之 标记 来 定义 ,使 用 
VALUE 属性 来 给 选项 指定 值 ,这 个 值 不 会 显示 在 浏览 器 中 。 如 果 需 要 指定 一 个 默认 的 
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选项 ,可 以 在 该 选项 中 加 上 一 个 SELECTED 属性 。 而 在 OPTION 二 一 /OPTION 二 之 
间 , 可 以 定义 显示 到 浏览 器 中 的 内 容 。 

如 果 需 要 定义 多 选 列表 ,可 以 通过 指定 二 SELECT 标记 的 一 个 属性 MULTIPLE 
来 完成 ,并 且 同 时 将 SIZE 属性 值 设 置 为 2 或 者 2 以 上 。 

示例 4-1 是 一 个 单 选 列表 和 多 选 列表 的 表单 元 素 用 法 的 例子 。 

闪 示 例 4-1 单 选 列表 和 多 选 列表 


<HTML> 
<BODY> 
<FORM NAME= "FN" ACTION=""> 
<SELECT NAME= "Favorite" SIZE="4" MULTIPLE> 
<OPTION VALUE= "Basketball"> 篮 球 < /OPTION> 
<OPTION VALUE= "Volleyball"> 排 球 < /OPTION> 
<OPTION VALUE="Table Tennis"> 后 乓 球 < /OPTION> 
<OPTION VALUE="Tennis"> 网 球 </OPTION> 
</SELECT> 
<BR><BR> 
<SELECT NAME= "Gender"> 
<OPTION VALUE= "Man" SELECTED> 男 < /OPTION> 
<OPTION VRLUE= "Woman"> 女 </OPTION> 
</SELECT> 
</FORM> 
< /BODY> 
</HTML> 


图 4-1 是 这 两 个 列表 的 显示 效果 ,第 一 个 为 多 选 列表 ,第 二 个 为 单 选 列表 。 在 第 二 个 
单 选 列表 中 ,指定 了 默认 选项 为 第 一 个 选项 。 


5. 单 选 框 
单 选 框 提供 给 用 户 多 选 一 的 组 件 , 它 的 基本 格式 如 下 : 


<INPUT TYPE="RADIO" NAME="#" VALUE="#"># 


单 选 框 也 是 使 用 二 INPUT 二 > 标记 来 定义 ,但 是 需要 
将 它 的 TYPE 属性 设置 为 RADIO ,需要 给 它 指定 一 个 名 
字 。 一 个 <INPUT>> 只 能 定义 一 个 单 选 框 选项 ,因此 对 “大 列 四 计 导 居 大 RE 
于 一 组 的 选项 ,必须 针对 每 个 选项 定义 并 且 需 要 给 它们 ”图 41 示例 41 的 显示 效果 
提供 一 样 的 名 字 , 这 样 ,这 些 选项 就 能 组 成 一 个 “组 ”。 在 
这 些 选项 中 ,每 次 最 多 只 能 有 一 个 选项 被 选中 。 如 果 需 要 指定 默认 的 选项 ,可 以 给 该 选项 
指定 CHECKED 属性 ,如 : 


<INPUT TYPE="RADIO" NAME=" 性 别 " VALUE=" 男 "> 男 
<INPUT TYPE="RADIO" NAME= "性别 " VALUE=" 女 " CHECKED> 女 


这 样 ,对 于 “性 别 ” 单 选 框 , 它 有 两 个 选项 ,默认 选项 是 “ 女 ”。 
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6. 多 选 框 
多 选 框 提供 给 用 户 一 个 选择 多 个 选项 的 组 件 , 它 的 基本 格式 如 下 : 


<INPUT TYPE="CHECKBOX" NAME="#" VALUE="#"># 


通常 情况 下 ,将 一 组 同样 性 质 的 多 选 框 指定 用 一 样 的 名 字 。 如 果 需 要 指定 某 些 选项 
为 默认 选项 ,可 以 给 这 个 选项 指定 CHECKED 属性 。 被 选择 中 的 各 个 选项 值 会 组 成 一 个 
字符 串 并 被 发 送 到 服务 器 端 , 各 个 选项 值 之 问 用 逗号 隔 开 。 

下 面 是 一 个 多 选 框 和 单 选 框 的 表单 元 素 用 法 的 例子 。 

闪 示 例 4-2 单 选 框 和 多 选 框 


<HTML> 
<BODY> 
< FORM NAME="FN" ACTION=""> 
选择 性 别 : 
<INPUT TYPE= "RADIO" NAME=" 性 别 " VALUE= " 男 "> 男 
<INPUT TYPE= "RADIO" NAME=" 性 别 " VALUE=" 女 " CHECKED> 女 
<BR><BR> 
选择 你 的 兴趣 爱好 : 
<INPUT TYPE= "CHECKBOX" NAME= "兴趣 " VALUE= "上 网 " CHECKED> 上 网 
<INPUT TYPE= "CHECKBOX" NAME= "兴趣 " VALUE= " 踢 球 "> 踢 球 
<INPUT TYPE= "CHECKBOX" NAME= "兴趣 " VALUE= "篮球 "> 篮球 
<INPUT TYPE= "CHECKBOX" NRME= "兴趣 " VALUE= "健身 " CHECKED> 健 身 
< /FORM> 
< /BODY> 
</HTML> 


图 4-2 是 它们 的 显示 效果 。 


选择 性 别 ，C 男 @ 女 
选择 你 的 兴趣 爱好 ， 风 上 网 三 踢 球 后 篮球 网 健身 


图 4-2 示例 4-2 的 显示 效果 


7. 按钮 


在 HTML 中 ,有 三 种 类 型 的 按钮 : Submit、Reset 和 Button。 使 用 Submit 按钮 可 以 将 表 
单 提交 到 FORM 标记 的 ACTION 所 指定 的 URL 中 ;Reset 按钮 可 以 将 表单 的 内 容 恢复 到 
原始 的 状态 ;而 Button 类 型 的 按钮 通常 情况 下 需要 和 JavaScript 结合 起 来 使 用 才 有 意义 。 

。 Submit 

Submit 按钮 定义 的 基本 格式 如 下 : 
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<INPUT TYPE="SUBMIT" NAME="#" VALUE= "#"> 


可 以 看 出 ,Submit 按钮 的 定义 也 是 使 用 二 INPUT 才 > 标记, 只 是 需要 将 TYPE 属性 值 
指定 为 SUBMIT 就 可 以 了 ,另外 需要 给 它 指定 一 个 名 字 。VALUE 属性 将 显示 在 
Submit 按钮 上 ,但 它 不 是 必需 的 ,如 果 没 有 指定 它 , 那 么 在 浏览 器 中 就 会 自己 加 上 一 个 
值 , 在 不 同 的 浏览 器 中 可 能 会 有 所 不 同 , 如 在 英文 的 正 中 通常 会 是 Submit Query', 而 在 
中 文 IE 中 是 “提交 查询 内 容 ” 等 。 

。 Reset 

Reset 按钮 用 于 将 表单 中 的 各 个 组 件 的 值 恢复 到 最 初 设置 的 初 值 , 它 的 基本 格式 如 下 : 


<INPUT TYPE="RESET" NAME="#" VALUE="#"> 


RESET 按钮 的 定义 也 使 用 二 INPUT 二 > 标记, 只 是 需要 将 TYPE 属性 值 指定 为 
RESET 就 可 以 了 ,另外 需要 给 它 指定 一 个 名 字 。VALUE 属性 将 会 显示 在 Reset 按钮 
上 , 它 可 以 不 指定 ,而 是 让 浏览 器 自己 给 它 赋值 ,不 同 的 浏览 器 中 会 有 一 些 差别 ,如 在 英文 
IE 中 显示 为 Reset, 而 在 中 文 正 中 显示 为 “ 重 置 "等 。 

。 Button 

Button 通常 需要 和 JavaScript 结合 起 来 才 起 作用 ,因为 它 本 身 不 能 完成 任何 的 功能 ， 
它 的 格式 如 下 : 


<INPUT TYPE="BUTTON" NAME="#" VALUE= "#"> 


与 Submit 和 Reset 按钮 一 样 ,Button 使 用 二 INPUT 才 来 定义 。 在 Button 定义 中 ， 
需要 给 VALUE 指定 一 个 值 ; 否 则 ,在 浏览 器 中 的 按钮 组 件 上 不 会 有 任何 的 文字 指示 。 

在 FORM 中 ,通常 至 少 需要 定义 一 个 Submit 按钮 ,这 样 表 单 才 可 以 提交 到 ACTION 指 
定 的 URL 中 。 当 然 ,使 用 Button 按钮 和 JavaScript 的 组 合 也 能 达到 一 样 的 目的 。 


要 求 登录 页 中 含有 : 

。 一 个 文本 框 ,用 于 输入 用 户 名 ; 

。 一 个 密码 框 ,用 于 输入 密码 ; 

。 含有 一 个 “提交 ”按钮 ,按钮 上 显示 “登录 ”; 

。 含有 一 个 指向 register. jsp 的 按钮 ,按钮 上 显示 “注册 ”; 

。 当 单 击 “ 提 交 ” 按 钮 时 ,表单 数据 提交 给 loginCheck. jsp 处 理 。 


闪 示 例 4-3 ”登录 页 login. jsp 


<center> 
<FORM name= "forml" method= "post" action=" "></P> 
<P> 用 户 名 < INPUT type="text" name="username" '></P> 
<P> 密 码 <INPUT type="password" name="password"></P> 
<P><INPUT type="submit" name="submit" value=" 提 交 " /> 
<INPUT type="button" name="button" value=" 注 册 " 


onclick="document.location.href='register.jsp'" /> 
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将 示例 4-3 中 的 login. jsp 复制 到 Tomcat 的 根 目录 下 ,启动 Tomcat 服务 器 ,在 浏览 
器 的 URL 中 输入 http://localhost:8080/login. jsp ,会 看 到 如 图 4-3 所 示 的 执行 效果 。 


图 4-3 login. jsp 的 显示 效果 


课本 开本 制作 用 户 注册 页 
要 求 注册 页 中 含有 如 下 内 容 : 
。 一 个 文本 框 ,用 于 输入 用 户 名 ; 
。 一 个 密码 框 ,用 于 输入 密码 ; 
。 一 组 单 选 按钮 ,用 于 选择 性 别 ; 
。 一 个 文本 区 域 ,用 于 输入 用 户 的 自我 介绍 ; 
一 组 单 选 列表 菜单 ,用 于 选择 用 户 所 在 的 学 院 ; 
一 组 复 选 框 ,用 于 用 户 选择 爱好 ; 
。 一 个 “提交 ”按钮 ,按钮 上 显示 为 “注册 ”; 
。 当 单 击 “ 提 交 ” 按 钮 时 ,表单 数据 提交 给 dealRegister. jsp 处 理 。 


闪 示 例 4-4 登录 页 register. jsp 


<form id="forml" name="forml" method="post" action="dealRegister.jsp"> 
<p><label>username 
<input type="text" name="username" /> 
</label>< /p> 
<p><label>password 
<input type="password" name="password" /> 
</label>< /p> 
<p><1label>gender male 
<input type="radio" name="gender" value= "male" /> 
</label> 
<label>female 
<input type="radio" name="gender" value="female" /> 
</label></p> 
<p> 
<label>introduction 
<textarea name="introduction" rows="6"></textarea> 
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</label>< /p> 
<p> 
<label>school 
<select name="comefrom"> 
<option>Computer Sciences</option> 
<option>Mechatronic Engineering< /option> 
<option>Physical Science< /option> 
</select> 
</label></p> 
<p> 
<label>hobby 
music <input type="checkbox" name="checkbox" value= "music" /> 
</label> 
<label> 
sports <input type="checkbox" name="checkbox" value="sports" /> 
</label> 
<label> 
dancing <input type="checkbox" name="checkbox" value="dancing" /> 
</label></p> 
<p> 
<label> 
<input type="submit" name="Submit" value="register" /> 
</label></p> 
</form> 


示例 4-4 的 显示 效果 如 图 4-4 所 示 。 


GO- htp://localhost a080/registerjsp =] [s(x 


introduction 
school [Computer Sciences 


hobby music 厂 sports T dancing 三 


图 4-4 register. jsp 的 显示 效果 


8. 隐藏 域 


隐藏 域 可 以 定义 在 FORM 中 ,用 来 传递 不 需 用 户 输入 的 值 。 与 其 他 的 FORM 组 件 
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不 同 , 它 不 会 显示 在 浏览 器 中 ,用 户 不 能 去 修改 它 的 值 。 它 的 定义 基本 格式 如 下 : 


<INPUT TYPE= "HIDDEN" NAME="#" VALUE=""> 


与 文本 框 等 组 件 一 样 ,隐藏 域 的 定义 也 是 使 用 二 INPUT 二 标记 来 完成 的 ,需要 将 它 
的 TYPE 属性 值 设置 为 HIDDEN ,并 且 需 要 给 它 指定 一 个 名 字 。 另 外 因为 隐藏 域 不 能 接 
收 用 户 的 输入 ,所 以 通常 需要 给 它 指定 一 个 VALUE 值 。 


9. 文件 上 载 组 件 


有 时 候 Web 应 用 需要 将 客户 端的 文件 上 载 到 服务 器 端 ,这 时 候 就 需要 使 用 文件 上 载 
组 件 来 接收 需要 上 载 的 文件 的 路 径 , 它 的 基本 格式 如 下 : 


<INPUT TYPE="FILE" NAME="#" MAXLENGTH="#" SIZE="#"> 


文件 上 载 组 件 也 使 用 二 INPUT 二 标记 来 定义 ,并 且 将 它 的 TYPE 属性 值 设 置 为 
FILE, 同 时 需要 给 它 的 NAME 选项 指定 一 个 值 。MAXLENGTH 和 SIZE 属性 的 含义 和 
文本 框 的 含义 一 样 。 文 件 上 和 载 组 件 在 浏览 器 上 的 表现 形式 为 一 个 “文本 框 ”" 和 一 个 “ 按 
钮 ”, 这 个 按钮 会 在 不 同 的 浏览 器 上 面 显示 类 似 “ 浏 览 ...” 的 内 容 。 单 击 这 个 按钮 ,将 打开 
一 个 文件 选择 对 话 框 ,让 用 户 选择 一 个 文件 。 

另外 ,如 果 要 让 文件 能 够 顺利 上 载 ,需要 将 对 应 的 FORM 的 ENCTYPE 属性 设置 成 
multipart/form-data, 并 且 一 定 需要 将 METHOD 属性 设置 为 post。 

在 示例 4-4 的 “提交 ”按钮 前 添加 如 下 的 文件 上 载 组 件 。 


<p><label>picture <input type="file" name="picture"></label></p> 


图 4-5 为 添加 了 文件 上 载 组 件 后 的 注册 页 在 IE 上 的 显示 效果 。 


TO je hp/ocahosta000reoistersp 3| x) [EF sans 四 日 


帘 妾 ”起 httpy/localhostsosorregister- 丛 ~ 园 ~ 山 ~ 四 FEpv 千 IRolv” 


school |Computer Sciences 。 


hobby music sports dancing 三 
piare — Ei 
| 


[| || Umaga ER ee 


图 4-5 文件 上 载 组 件 后 的 显示 效果 
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4.2 JSP 语 法 


JSP(Java Server Pages) 是 由 Sun Microsystems 公司 倡导 、 许 多 公司 参与 并 共同 创建 
的 一 种 使 软件 开发 者 可 以 响应 客户 端 请 求 ,而 动态 生成 HTML、XML 或 其 他 格式 文档 的 
Web 网 页 的 技术 标准 。JSP 技术 是 以 Java 语言 作为 脚本 语言 的 ,JSP 网 页 为 整个 服务 器 
端的 Java 库 单元 提供 了 一 个 接口 来 服务 于 HTTP 的 应 用 程序 。JSP 网 页 (* .jsp) 就 是 
在 传统 的 HTML 文件 (* . htm,* .html) 中 加 入 Java 程序 片段 和 标记 而 构成 的 。 


4.2.1 JSP 页 面 的 组 成 


要 想 在 JSP 页 面 中 实现 丰富 的 动态 功能 ,需要 JSP 各 种 组 成 成 分 的 帮助 ,常用 的 组 
成 成 分 及 功能 说 明 见 表 4-2。 
表 4-2 主要 JSP 页 面 组 成 


元 素 描 述 

注释 在 客户 端 建立 一 个 可 见 或 不 可 见 的 注释 
声明 在 脚本 语言 里 声明 变量 和 方法 

表达 式 在 脚本 语言 里 加 入 表达 式 

程序 片 在 脚本 语言 里 加 入 脚本 程序 段 

包含 指示 加 入 一 个 用 来 说 明 JSP 基本 要 素 的 文件 

页 面 指示 定义 整个 JSP 页 面 的 属性 

标签 指示 在 JSP 页 面 里 定义 标签 库 和 定制 标签 的 前 级 


4.2.2 JSP 注释 

JSP 注释 可 以 大 致 分 为 两 种 : 一 种 是 可 以 发 送 到 客户 端的 ,一 种 是 不 会 发 送 给 客户 
端 而 仅 提 供给 JSP 程序 员 阅 读 的 ,这 两 种 注释 在 语法 上 是 不 同 的 。 

1. 发 送 到 客户 端的 JSP 注释 

发 送 到 客户 端的 JSP 注释 的 基本 语法 格式 如 下 : 

<!-- 注 释 [<%= 表 达 式 8>]--> 


这 种 方式 其 实 是 HTML 注释 方式 ,只 不 过 可 以 在 里 面 加 上 JSP 表达 式 , 动 态 生成 注 
释 内 容 。 
闪 示 例 4-5 ”发 送 到 客户 端的 注释 jspCommentsl.jsp 


<%@page contentType="text/html;charset=gb18030"%> 
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<HEAD> 

<TITLE> 发 送 到 客户 端的 注释 < /TITLE> 
< /HEAD> 
<BODY> 


<!-- 这 个 注释 会 发 送 到 客户 端 <s=new java.util.Date()%>--> 
</BODY> 


其 中 “二! 一 一 ”和 “一 一 二 ”部 分 内 容 不 会 在 浏览 器 中 显示 ,如 图 4-6 所 示 。 


TO [8 pywocanoseosokpcomnenslp 蝇 图 网 [sseaoma 加 


从 ~ 园 - 必 "四 mm 对 IRO” 


图 4-6 示例 4-5 的 执行 结果 


使 用 浏览 器 查看 源 文件 的 功能 ,可 以 看 到 如 图 4-7 的 结果 ,表明 “二 ! 一 一 "和 "一 一 二 ” 
部 分 的 内 容 已 经 发 送 到 了 客户 端 。 


HEAD> : 
| 《TITLE> 发 送 到 客户 端的 注释 5/TITLE> 


《4! 一 这 个 注释 会 发 送 到 客户 端 Jon May 20 18:35:04 CST 2013 一 > 


一 


图 4-7 在 浏览 器 中 查看 页 面 源 文件 的 结果 (一 ) 


2. 不 发 送 到 客户 端的 注释 


不 发 送 到 客户 端的 注释 的 基本 格式 如 下 : 
<%- -注释 内 容 --%> 


在 “到 %% 一 一 ”和 ”一 一 外 >? 部 分 的 内 容 只 是 为 方便 程序 员 或 其 他 相关 人 员 阅 读 代码 
而 添加 的 注释 ,不 会 被 编译 ,不 会 被 执行 ,也 不 会 被 发 送 到 客户 端 。 示 例 4-6 中 的 注释 语 
句 就 是 这 种 注释 。 

闪 示 例 4-6 jspComments2. jsp 


<%Q@page contentType="text/html;charset=gb18030"%> 
<HTML> 
<HEAD> 

<TITLE> 不 发 送 到 客户 端的 注释 < /TITLE> 


</HEAD> 
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<BODY> 
<&%-- 这 个 注释 不 会 发 送 到 客户 端 <s=new java.util.Date ()%>--%> 
</BODY> 


示例 4-6 运行 后 ,在 客户 端 浏览 器 中 查看 它 的 源 文件 ,内 容 如 图 4-8 所 示 。 


图 4-8 在 浏览 器 中 查看 页 面 源 文件 的 结果 (二 ) 


可 以 看 出 ,这 个 注释 没有 发 送 到 客户 端 。 

因为 在 JSP 程序 中 ,有 一 部 分 是 Java 代码 ,因此 ,除了 上 面 两 种 JSP 的 注释 方式 以 
外 ,在 Java 代码 中 ,还 可 以 像 普通 的 Java 类 中 一 样 使 用 Java 的 注释 语句 ,单行 注释 用 
“//”, 而 多 行 注释 使 用 */ x* ”和 “x* /包含 起 来 。 推 荐 使 用 "// ”的 注释 方式 ,因为 在 某 些 集 
成 开发 环境 如 MyEclipse 中 ,这 种 注释 方式 会 使 用 绿色 等 颜色 特别 显示 ,使 开发 人 员 容易 
将 注释 与 代码 的 其 他 部 分 区 分 开 来 。 


4.2.3 JSP 程序 片 


如 果 业 务 逻 辑 比 较 复 杂 ,JSP 程序 中 可 能 含有 大 段 的 Java 代码 ,这 时 就 要 使 用 Java 
程序 片 。Java 程序 片 就 是 用 "到 办 ”和 "名 盖 "包含 起 来 的 Java 代码 段 。 格 式 如 下 : 

<% 

Java 代码 

多 > 

在 JSP 中 ,可 以 在 任何 地 方 插入 任何 数量 的 Java 程序 片 。 但 要 注意 的 是 ,Java 程序 
片 之 间 不 能 由 套 。 


课堂 组 可 下 3 java 程序 片 的 使 用 
要 求 : 将 1 一 10 累加 的 结果 显示 在 下 浏览 器 中 。 
提示 : JSP 中 的 Java 程序 片 包含 在 所 台中 > 标签 中 。 


闪 示 例 4-7 Java 程序 片 实例 doSum. jsp 
<HTIML> 


<HEAD> 
<TITLE>1-10 的 累加 < /TITLE> 
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</HEAD> 
<BODY> 
<% 
int sum=0; 
for (int i=1;i<=10;i++) 


{ 一 Java 程 序 片 


sum+=i; 


Out .Print (sum); 
多 > 


</BODY> 
</HTML> 


示例 中 的 out. print(sum) 的 作用 是 向 浏览 器 中 输出 sum 的 值 ,out 是 一 个 内 置 对 象 。 
关于 jsp 中 的 内 置 对 象 ,将 在 后 续 小 节 中 学 习 。 
图 4-9 是 这 个 程序 的 运行 结果 。 


Ge = [大 htpy/ocalhostaoeo/dosumjsp 。” 习 回 |] 


窗 灾 项-l1ohM0xn0 


图 4-9 ”doSum. jsp 的 运行 结果 


4.2.4 JSP 声明 


在 JSP 中 ,方法 的 声明 以 及 全 局 变量 的 声明 应 该 放 在 JSP 声明 部 分 , 它 的 基本 格式 
如 下 : 


<$! 声 明 %$> 


JSP 声明 中 的 变量 是 全 局 变量 ,所 以 它 可 以 在 任何 地 方 被 使 用 。 
下 面 来 看 一 个 JSP 声明 的 例子 。 
闪 示 例 4-8 JSP 声明 效果 演示 declaration. jsp 


<HTML> 

<HEAD> 

<TITLE>declaration </TITLE> 

</HEAD> 

<BODY> 
<31 一 全 局 声明 ] 
private String str="This is globalvariable"; 
private int sum(int n) 


{ 
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int result=0; 
for(int i=1;i<=n;i++) 
{ 
result+=i; 
} 


return result; 


< 
int s=sum(10); 
out .print ("The result is :"+s); 
名 > 
<hr> 
<%=str$%> 
</BODY> 


这 个 例子 在 JSP 声明 中 定义 了 一 个 全 局 变量 str 和 一 个 方法 sum () ,这 个 方法 完成 
用 户 从 1 到 n 的 累加 。 定 义 方法 后 ,可 以 在 java 程序 片 中 直接 调用 这 个 方法 ,或 者 可 以 
在 另 一 个 方法 声明 中 调用 它 。 

图 4-10 是 这 个 程序 的 运行 结果 。 


J [=] E3 


TSO | tp//ocahostaos0/dedaraionjsp 可 疾风 [Be sans [Ea 


从 ~ 园 - 山 ~ 四 FEplv 和 对 IIROv” 


图 4-10 declaration. jsp 运行 结果 


4.2.5 JSP 表达 式 


JSP 表达 式 的 基本 语法 如 下 : 
<%=expression%®%> 


其 中 , expression 是 一 个 可 以 转换 成 字符 串 的 表达 式 。JSP 表达 式 的 作用 是 将 
expression 的 运算 结果 输出 到 客户 端 。 使 用 表达 式 有 几 个 地 方 需要 注意 : 

(1) JSP 表达 式 只 能 单独 出 现 ,也 就 是 说 ,不 能 在 一 对 “二 %” 和 “% 二 ”之 间 出 现 多 条 
语句 ; 

(2) JSP 表达 式 和 普通 的 Java 语句 不 同 ,不 能 用 “ ; ”结束 ; 

(3) JSP 表达 式 的 “二 %”“% 记 ”和 expression 只 能 在 同一 行 上 ; 

(4) 表达 式 expression 必须 能 够 转换 成 字符 串 形式 ; 

(5) 不 在 表达 式 中 定义 变量 和 方法 ; 

(6) 不 能 在 表达 式 中 使 用 返回 值 为 void 的 方法 。 
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使 用 表达 式 的 效果 和 在 Java 程序 片 中 使 用 out. println() 的 效果 是 一 样 的 ,它们 都 会 
向 客户 端 输出 内 容 。 事 实 上 ,在 JSP 中 ,也 可 以 使 用 一 个 内 置 的 out 对 象 来 实现 这 个 功 
能 。 关 于 out 的 使 用 ,请 读者 参考 后 续 章 节 的 内 容 。 


党 襄 红 本 4 动态 输 员 表格 内 容 


要 求 : 在 一 个 三 行 两 列 的 表格 中 输出 5、10、15 的 累加 值 。 


5 15 
10 55 
15 120 


提示 : 
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。 全 局 变量 声明 和 表达 式 的 使 用 ,方便 实现 动态 内 容 按 设计 者 的 要 求 出 现在 网 页 的 


不 同 部 位 。 
。 Html 标签 和 Java 程序 片 可 交替 出 现 ,但 Html 标签 不 能 被 包含 在 Java 程序 片 内 。 


闪 示 例 4-9 JSP 表达 式 效 果 演 示 expression. jsp 


<table border= "1"> 
< 
int sum=0; 
for(int j=1;j<=3;j++){ 
for (int i=1;i<=j* 5;i++) 
{ 
sumt=i; 


} 


各 > 兆 、 
a JSP 表 达 式 的 使 用 


<td><%=j* 5%$>< /td> 
<td><%= Sumgs>< /td> 
< /tr> 
<% 
sum=0; 
} 
$> 
</table> 


示例 4-9 的 显示 效果 如 图 4-11 所 示 。 


GO [8 pocanorooon/eroressionisn | Gj] Ee"F. su |[E 问 
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图 4-11 expression. jsp 的 显示 效果 


4.3 JSP 指令 标签 


JSP 指令 标签 主要 用 来 提供 整个 JSP 网 页 相关 的 信息 ,并 且 用 来 设 定 JSP 页 面 的 相 
关 属 性 。JSP 指令 的 语法 格式 如 下 : 


<gQ@directive {attr=value} %> 


directive 为 标签 的 名 称 。JSP 有 三 个 指令 标签 : page、include 和 taglib ,本 书 只 介绍 
include 和 page 指令 。taglib 指令 用 于 指定 需要 用 到 的 标记 库 。attr 部 分 为 指令 标签 的 
属性 。 


4.3.1 page 指令 


page 指令 允许 程序 员 导 入 需要 的 类 、 指 明 JSP 输出 内 容 类 型 .控制 session 等 。 其 语 
法 格式 如 下 : 


<%@page {attr=value} $%> 
下 面 介绍 page 指令 的 主要 属性 。 
1. contentType 属性 


contentType 属性 用 于 指明 JSP 输出 的 内 容 的 MIME(Multipurpose Internet Mail 
Extensions) 类 型 。 在 默认 情况 下 ,JSP 的 MIME 类 型 是 “text/html; charset 王 ISO-8859- 
1”, 中 文 网 页 常见 的 MIME 类 型 有 “text/html; charset 二 GB2312”、“text/html;charset 二 
GBK"” 或 者 “text/html;charset 二 GB18030”。 


= 党 营 红 可 5 解决 JSP 页 而 的 中 文 筷 码 问 题 
要 求 : 图 4-9 左上 角 的 网 页 标题 中 出 现 了 中 文 乱 码 ,将 这 一 问题 更 正 ,使 中 文正 常 
提示 : 在 page 指令 中 设置 contentType 来 指定 本 页 采用 中 文 简体 编码 集 。 


将 下 面 的 page 指令 添加 到 doSum. jsp 的 第 一 行 。 


<s%@page contentType="text/html;charset=gb18030" %> 
修改 后 的 doSum. jsp 的 显示 效果 如 图 4-12 所 示 。 与 图 4-9 相 比 ,可 以 看 到 已 经 更 正 
了 页 面 标题 的 乱码 问题 。 
2. pageEncoding 属性 
通过 contentType 可 以 设置 JSP 输出 结果 的 MIME 类 型 ,如 果 只 想 改 变 文件 的 编码 
(使 用 默认 的 text/html 内 容 类 型 ) ,那么 只 需要 使 用 pageEncoding 就 可 以 了 ,代码 如 下 : 
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页 面 标题 不 再 是 中 文 乱码 


二 加 国 国 图 国 国 2 EE 


图 4-12 更 正 了 中 文 乱码 后 的 显示 效果 


<%@page pageEncoding= "gb18030" $%> 


3. import 属性 


如 果 编 写 过 Java 程序 的 读者 都 应 该 知道 ,在 Java 程序 中 ,通常 需要 用 到 JDK 中 的 其 
他 类 ,这 个 时 候 , 就 需要 将 这 个 类 导入 (import) ,那么 ,既然 在 JSP 中 也 需要 编写 Java 代 
码 ,应 该 如 何 导 入 其 他 的 类 呢 ? 

通过 page 指令 的 import 属性 ,可 以 方便 地 将 需要 用 到 的 类 引入 进来 , 它 的 用 法 
如 下 : 


<%Q@page import="package.Class"s> 
或 者 
<%@page import="package. * "%> 
如 果 想 要 引入 多 个 包 中 的 类 ,可 以 使 用 下 面 的 方法 : 


<%@page import="packagel.Class"%®> 
<%@page import="package2.Class"%®> 


或 者 
<%@page import="package.Classl, package2.Class2"%> 
提示 : import 是 page 指令 中 唯一 一 个 可 以 在 同一 个 JSP 中 多 次 出 现 的 属性 。 
4. session 属性 


session 属性 用 于 控制 页 面 是 否 需 要 使 用 session( 会 话 ) ,这 个 属性 值 可 以 为 true 或 
者 false。 让 true, 表 示 这 个 页 面 需要 使 用 session; 如 果 属 性 值 为 false， 
则 表示 这 个 页 面 不 能 使 用 session。 下 面 是 page 指令 中 的 session 属性 设置 的 例子 。 


<%@page session="false" $> 


使 用 了 上 面 的 page 指令 的 JSP 页 面 不 能 访问 session, 或 者 为 用 户 创建 新 的 session。 
关于 JSP 中 的 session ,将 会 在 后 面 的 小 节 中 继续 学 习 。 


52 


5. errorPage 和 isErrorPage 属性 


errorPage 用 于 指明 在 这 个 JSP 页 面 中 出 现 未 被 捕获 的 异常 时 , 跳 转 到 哪个 页 面 来 处 
理 。 通 常 , 跳 转 的 页 面 需要 使 用 isErrorPage 来 指明 可 以 用 于 其 他 页 面 的 错误 处 理 。 
errorPage 的 用 法 如 下 : 


<%Q@page errorPage="errorHandle.jsp" $> 

通过 上 面 的 page 指令 ,可 以 指明 当 这 个 JSP 中 出 现 错误 的 时 候 ,会 跳 转 到 
errorHandle. jsp 来 处 理 。 在 errorHandle. jsp 中 ,通常 需要 使 用 isErrorPage 属性 来 说 明 
它 可 以 用 于 其 他 页 面 的 错误 处 理 ,代码 如 下 : 

<%@page isErrorPage="true" %> 

在 错误 处 理 页 面 中 可 以 使 用 内 置 的 exception 对 象 来 获得 相关 的 异常 信息 ,例如 使 用 
它 的 printStackTrace( ) 方 法 来 打印 异常 堆栈 ,如 果 在 错误 处 理 页 面 中 使 用 了 exception 
内 置 对 象 ,那么 ,必须 在 这 个 jsp 页 面 中 使 用 page 指令 的 isErrorPage 属性 ,并 且 将 它 的 
值 设 置 成 true, 否 则 将 无 法 使 用 exception 这 个 内 置 对 象 。 关 于 exception 内 置 对 象 的 更 
多 知识 ,将 在 后 续 内 容 中 介绍 。 


6. buffer 属性 


buffer 属性 用 于 指定 在 使 用 out 内 置 对 象 向 客户 端 输出 内 容 的 时 候 使 用 的 缓冲 区 的 
大 小 , 它 的 默认 值 是 8KB。 


7. info 属性 


info 属性 可 以 用 于 指定 对 这 个 JSP 页 面 的 一 些 说 明 信 息 ,可 以 通过 Servlet 的 
getServletInfo() 方 法 获得 。 它 的 用 法 如 下 : 


<%@page info="Jsp Message"®> 


8. isThreadSafe 属性 


isThreadSafe 属性 用 于 控制 JSP 页 面 是 否 允 许 并 行 访问 , 它 可 取 两 个 值 之 一 : true 
或 者 false, 默 认 是 true, 表 示人 允许 并 行 访问 。 在 JSP 2. 0 中 ,这 个 属性 不 推荐 使 用 ,或 者 
说 ,不 推荐 将 它 设置 成 false。 对 于 需要 控制 并 发 访问 的 地 方 , 应 该 在 代码 中 实现 ,不 能 依 
赖 于 将 isThreadSafe 设置 成 false。 


9. isELIsgnored 属性 


该 属性 是 JSP 2. 0 中 新 引入 的 一 个 属性 , 它 所 要 控制 的 是 ,忽略 还 是 处 理 JSP 2.0 中 

的 EL(Expression Language) 。 如 果 将 这 个 值 设 置 成 true:, 那 么 ,将 忽略 EL; 如 果 将 它 设 

置 成 false, 将 对 EL 进行 正常 的 运算 。 需 要 注意 的 是 ,在 只 支持 JSP 1. 2 的 服务 器 中 ,不 
能 使 用 这 个 属性 。 
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10. language 属性 


language 属性 用 于 指定 JSP 所 使 用 的 脚本 语言 ,在 目前 的 情况 下 ,只 有 一 种 选择 , 那 
就 是 Java。 


11. extends 属性 


使 用 这 个 属性 可 以 指定 JSP 所 转换 成 的 Servlet 的 父 类 。 
4.3.2 include 指令 


include 指令 用 于 将 多 个 JSP 页 面 (或 者 HTML、XML 文件 ) 组 合 起 来 ,从 而 成 为 一 
个 “完整 "的 JSP 页 面 。 它 的 基本 语法 格式 如 下 : 


<%@include file="jspFile"%®%> 


这 个 指令 有 一 个 必要 的 属性 file, 它 用 于 指明 包含 哪个 文件 。 在 一 个 JSP 中 ,可 以 使 
用 多 个 include 指令 来 包含 多 个 JSP 或 者 HTML 文件 。 

由 于 被 引入 的 文件 的 内 容 将 在 include 指令 的 位 置 被 引入 ,并 与 引用 的 文件 拼合 成 为 
一 个 文件 后 再 进行 编译 ,所 以 被 引入 的 文件 中 不 要 出 现 二 html 之 一 /html 之 一 body>> 
二 /body 二 等 html 标签 ,以免 破坏 整个 网 页 的 html 标签 结构 。 

在 Web 应 用 中 ,无 论 页 面 输出 什么 内 容 , 页 头 、` 页 脚 和 导航 栏 都 是 一 样 的 。 页 头 一 般 
是 网 站 的 Logo 图 片 ,而 页 脚 是 一 些 版 权 的 声明 等 。 如 果 只 使 用 静态 网 页 技术 ,就 需要 在 
这 些 页 面 中 写 人 相应 的 页 头 、 页 脚 和 导航 栏 的 HTML 代码 ,这 种 方式 会 带 来 很 大 的 维护 
问题 : 一 旦 这 些 内 容 需 要 进行 修改 ,就 需要 修改 所 有 的 包含 了 这 些 内 容 的 页 面 。 通 过 
JSP 的 include 指令 就 可 以 避免 这 一 问题 。 

课堂 练习 4-6 

要 求 用 include 指令 在 课堂 练习 4-1 和 4-2 实现 的 登录 页 和 注册 页 中 添加 导航 栏 。 

要 求 : 登录 页 和 注册 页 的 顶部 都 显示 同样 的 导航 栏 ,使 用 导航 栏 可 在 这 两 个 页 面 间 
自由 切换 。 

提示 : 将 主 菜单 制作 成 一 个 网 页 ,在 网 站 的 每 个 网 页 中 使 用 include 指令 引入 。 


首先 编写 导航 栏 文件 menu. jsp ,如 示例 4-10 所 示 。 
闪 示 例 4-10 导航 栏 文 件 menu. jsp 

<%Q@page pageEncoding= "gb18030"%> 

<center> 


<a href="login.jsp"> 登 录 </a> 
<a href="register.jsp"> 注 册 </a> 


在 login. jsp 和 register. jsp 中 添加 如 下 语句 : 
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兴 示 例 4-11 引入 导航 栏 的 login. jsp 


<sQ@page contentType="text/html;charset=GB18030" language="java"®> 
<HTML> 
<HERAD><TITLE> 登 录 页 </TITIE></HERD> 
<BODY align="center"> 
<%@ include file="menu.jsp"s> 
<table width="316" border= "1" align="center" bordercolor= "#666666" bgcolor= 
"CCCCCC"> 
<tr> 
<td width="306" bgcolor= "#CCCC99"> 
<table width="280" align="center" cellpadding="10"><tr align= 
"center"><td> 
<FORM name="forml" method="post" action="loginCheck.jsp"></P> 
<P> 用 户 名 <INPUT type= "text" name= "username"> 
<P> 密 码 <INPUT type="password" name="password"></P> 
<P><INPUT type="submit" name= "submit" value=" 提 交 " /> 
</P> 
< /FORM> 
</th></tr></table> 
</td></tr></table> 
</BODY> 
<HTML> 


示例 4-11 的 显示 效果 如 图 4-13 所 示 。 


SO- 轿 twpwocatostsoso0/o0injsp =] Ex [EF. rns ED 


但 ~ 园 -~ 蜗 ~ 四 IEp vv 对 IRov” 


图 4-13 引入 了 导航 栏 的 登录 页 


include 指令 的 优 缺 点 如 下 : include 指令 可 以 在 转换 阶段 就 将 对 应 的 文件 包含 进来 ， 
所 以 ,可 以 在 JSP 中 定义 在 主页 面 中 使 用 到 的 变量 等 。 这 是 include 指令 的 优点 。 但 是 ， 
它 有 一 个 很 大 的 缺点 , 当 被 包含 中 的 文件 内 容 发 生 改变 的 时 候 , 服 务 器 可 能 不 能 监测 到 ， 
因此 ,此 时 服务 器 不 会 对 它 进行 重新 转换 和 编译 ,这 可 能 会 带 来 很 大 的 维护 问题 ,因为 这 
个 时 候 需 要 改变 所 有 用 到 改动 的 被 包含 文件 的 JSP 文件 的 修改 日 期 ,这 样 才能 够 让 对 应 
的 JSP 文件 重新 转换 和 编译 。 
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4.4 ”JSP 动作 标签 


所 谓 JSP 动作 标签 ,是 在 JSP 中 利用 XML 语法 格式 的 标记 来 控制 Servlet 容器 的 
行为 。 
动作 标签 的 语法 格式 为 : 


<jsp:directive {attr=value}> 


JSP 动作 包括 如 下 内 容 。 

。jsp:include 在 页 面 被 请 求 的 时 候 引 入 一 个 文件 。 

。 jsp:useBean 获得 一 个 JavaBean 实例 。 

。 jsp:setProperty 设置 JavaBean 的 属性 。 

。 jsp:getProperty 获得 某 个 JavaBean 的 属性 。 

。jsp:forward 把 请 求 转 到 一 个 新 的 页 面 。 

。jsp:param 为 其 他 标签 提供 附加 信息 。 

。jsp:plugin 根据 浏览 器 类 型 为 Java 插件 生成 OBJECT 或 EMBED 标记 。 

本 节 主 要 讲解 include 和 forward 这 两 个 动作 ,而 useBean .setProperty .getProperty 
等 动作 将 在 第 7 章 讲 解 。plugin 动作 在 开发 中 用 得 很 少 ,不 再 讲解 ,请 读者 根据 其 他 的 动 
作 以 及 相关 文档 学 习 。 


4.4.1 include 动作 


include 动作 标签 也 能 起 到 引入 一 个 文件 的 作用 。 与 include 指令 标签 不 同 的 是 ， 
include 动作 标签 “告诉 ”JSP 页 面 动 态 地 包含 一 个 文件 , 即 当 JSP 引擎 把 JSP 页 面 转译 成 
Java 文件 时 ,不 把 JSP 页 面 中 动作 指令 include 所 包含 的 文件 与 原 JSP 页 面 合 并 成 一 个 
新 的 JSP 页 面 , 而 是 “告诉 ”Java 解释 器 ,这 个 文件 在 JSP 运行 时 才 包 含 进来 ,这 样 就 克服 
了 include 指令 的 缺点 。 

include 指令 的 语法 如 下 : 


<jsp:include page="includeFile"/> 
这 是 一 个 XML 语法 结构 ,所 以 这 个 标记 要 有 结束 符 。 它 的 另外 一 种 语法 方式 如 下 : 


<jsp:include page="includeFile"> 

</jsp:include> 

其 中 ,page 属性 用 于 指定 需要 包含 进来 的 资源 , 它 可 以 是 HTML 文件 JSP 输出 、 
Servlet 输出 等 。 它 只 会 将 被 包含 文件 中 的 输出 包含 进来 ,如 果 被 包含 文件 中 不 会 产生 输 
出 ,那么 使 用 这 个 include 动作 就 没有 什么 意义 。 

在 include 动作 中 ,除了 page 这 个 必需 的 属性 以 外 ,还 有 一 个 可 选 的 flush 属性 , 它 可 
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以 取 值 true 或 者 false。 这 个 属性 用 于 控制 是 否 在 页 面包 含 进来 之 前 清空 主页 面 的 输出 
流 ( 默 认为 false, 表 示 不 清空 )。 


4.4.2 ” forward 动作 


在 JSP 中 ,可 以 使 用 forward 动作 来 实现 程序 的 转向 ,例如 ,从 loginCheck. jsp 转向 
到 welcome. jsp 可 以 在 loginCheck. jsp 中 使 用 forward 动作 。 


<jsp:forward page="welcome.jsp"/> 


这 样 , 当 在 loginCheck. jsp 中 执行 forward 动作 的 时 候 , 它 将 会 跳 转 到 welcome. jsp。 
4.4.3 plugin 动作 


在 页 面 中 使 用 普通 的 HTML 标记 <applet>.…< 一 /applet> ,可 以 让 客户 端 下 载 并 运 
行 一 个 Java applet 小 应 用 程序 ,但 并 不 是 所 有 的 客户 的 浏览 器 都 支持 Java applet 小 程 
序 。 而 使 用 plugin 动作 标签 可 以 保证 客户 能 执行 小 应 用 程序 。 

过 jsp:plugin 记 为 Web 开发 人 员 提 供 了 一 种 在 JSP 文件 中 嵌入 客户 端 运行 的 Java 
程序 的 方法 。 一 般 来 说 ,一 jsp:plugin 之 元 素 会 指定 对 象 是 Applet 还 是 Bean, 同 样 也 会 
指定 class 的 名 字 及 位 置 ,另外 还 会 指定 将 从 哪里 下 载 这 个 Java 插件 。 

二 jsp:plugin 二 常用 属性 及 说 明示 例如 下 : 

<jsp:plugin type="applet|bean" code= "类 文件 名 "codebase= "类 文件 目录 " 


align= "bottom|ltoplmiddlelleftlright" height=" 插 件 显示 区 域 的 高 度 " 
width= "插件 显示 区 域 的 高 度 "> 


4.4.4 param 动作 


param 标签 以 “名 字 一 值 ”* 的 形式 为 其 他 标签 提供 附加 信息 。 这 个 标签 与 include、 
forward 和 plugin 动作 标签 一 起 使 用 。 

例如 下 面 的 用 法 ,可 以 向 跳 转 到 的 页 面 传送 信息 。 

<jsp:forward page="desFile"> 


<jsp:param name="paramName" value= "paramValue"/> 


</jsp:forward> 


4.4.5 JavaBean 相关 动作 标签 


< 一 jsp:useBean 二 : 用 来 装载 一 个 在 JSP 页 面 中 使 用 的 JavaBeans 。 
所 jsp:getProperty 二 : 用 来 获取 Bean 的 属性 值 并 将 其 转化 为 一 个 字符 串 ,然后 将 其 
插入 到 输出 页 面 中 。 
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过 jsp:setProperty 之 : 用 来 设置 .修改 Bean 中 的 属性 值 。 
这 些 标签 的 使 用 方法 将 在 第 5 章 中 结合 JavaBean 的 学 习 再 详细 讲解 。 


4.5 内 置 对 象 


JSP 的 内 置 对 象 在 JSP 中 非常 重要 ,这 些 内置 对 象 是 由 Web 容器 创建 出 来 的 ,所 以 
用 户 不 用 自己 创建 。JSP 内 置 对 象 是 Web 容器 加 载 的 一 组 类 , 它 不 像 一 般 的 Java 对 象 
那样 用 new 去 获取 实例 ,而 是 可 以 直接 在 JSP 页 面 中 使 用 的 对 象 。 

JSP 一 共有 9 个 内 置 对 象 ,如 表 4-3 所 示 。 
表 4-3 JSP 内 置 对 象 描述 


内 置 对 象 名 
request 


response 


session 


application 


实例 化 的 类 
javax. servlet. http. HttpServletRequest 


javax. servlet. http. HttpServletResponse 


javax. servlet. http. HttpSession 


javax. servlet. servletContext 


描 述 
该 对 象 封装 了 一 次 请 求 ,客户 端的 请 求 参数 
都 被 封装 在 该 对 象 里 
代表 服务 器 对 客户 端的 响应 
该 对 象 代表 一 次 会 话 。 从 客户 端 浏览 器 与 站 
点 建立 连接 起 即 开始 会 话 , 直 到 关闭 浏览 器 
时 结束 会 话 
该 实例 代表 JSP 所 属 的 Web 应 用 本 身 ,可 用 
于 JSP 页 面 或 者 在 Servlet 之 间 交 换 信息 
该 实例 代表 JSP 页 面 的 输出 流 ,用 于 输出 内 


out javax. servlet. jsp. jspWriter 容 , 从 而 形成 HTML 页 面 
| 该 对 象 代表 该 JSP 页 面 的 上 下 文 , 使 用 该 对 
pageContext | javax. servlet. jsp. pag' 象 可 以 访问 页 面 中 的 共享 数据 
page java. lang. Object 代表 该 页 面 本 身 
config javax. servlet. servletConfig 该 实例 代表 该 JSP 的 配置 信息 
该 实例 代表 其 他 页 面 中 的 异常 和 错误 。 只 有 
。 ea fai bl 当 页 面 是 错误 处 理 页 面 , 即 编译 指令 page 的 
exception java. lang. Throwable isErrorPage 属性 为 true 时 ,该 对 象 才 可 以 
使 用 
4.5.1 request 对 象 
客户 端的 请 求 信息 被 封装 在 request 对 象 中 ,通过 它 可 以 了 解 到 客户 的 需求 ,然后 做 
出 响应 。 它 是 HttpServletRequest 的 实例 。 
常用 方法 说 明 如 下 。 


(1) object getAttribute(String name) : 返回 指定 属性 的 属性 值 。 

(2) Enumeration getAttributeNames(): 返回 所 有 可 用 属性 名 的 枚 举 。 
(3) String getCharacterEncoding(): 返回 字符 编码 方式 。 

(4) int getContentLength(): 返回 请 求 体 的 长 度 ( 以 字 节 数 ) 。 
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(5) String getContentType() : 得 到 请 求 体 的 MIME 类 型 。 

(6) ServletInputStream getInputStream() : 得 到 请 求 体 中 一 行 的 二 进 制 流 。 

(7) String getParameter(String name): 返回 name 指定 参数 的 参数 值 。 

(8) Enumeration getParameterNames(): 返回 可 用 参数 名 的 枚 举 。 

(9) String[ ] getParameterValues(String name): 返回 包含 参数 name 的 所 有 值 的 


数组 


( 
( 
( 
( 
( 
( 
( 
( 


0) String getProtocol(): 返回 请 求 用 的 协议 类 型 及 版 本 号 。 

1) String getScheme(): 返回 请 求 用 的 计划 名 ,如 http、https 及 ftp 等 。 
2) String getServerName(): 返回 接受 请 求 的 服务 器 主机 名 。 

3) int getServerPort(): 返回 服务 器 接受 此 请 求 所 用 的 端口 号 。 

4) BufferedReader getReader(): 返回 解码 过 了 的 请 求 体 。 

5) String getRemoteAddr(): 返回 发 送 此 请 求 的 客户 端 IP 地 址 。 

6) String getRemoteHost(): 返回 发 送 此 请 求 的 客户 端 主机 名 。 

7) void setAttribute(String key,Object obj): 设置 属性 的 属性 值 。 

8) String getRealPath(String path) : 返回 一 虚拟 路 径 的 真实 路 径 。 
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在 课堂 练习 4-6 的 基础 上 ,进一步 模拟 用 户 登 录 的 功能 。 

要 求 : 当 用 户 输入 指定 的 用 户 名 和 密码 时 , 跳 转 到 欢迎 信息 页 welcome. jsp; 当 用 户 
名 和 密码 错误 时 , 跳 转 到 错误 提示 页 error. jsp。 

提示 : 判断 字符 串 sl 与 s2 是 否 相 等 用 sl. equals(s2) ,返回 值 为 布尔 型 true/false。 


闪 示 例 4-12 ”登录 处 理 页 loginCheck. jsp 


<% 


多 > 


< 对 


String uname=request .getParameter ("username"); 》 使 用 request 内 置 对 象 的 . 
String pass=request .getParameter ("password"); ee 


if(uname!=nullg&&pass!=null&&uname.equals ("liming") &&pass.equals ("aaa")) 
+ 


用 户 名 为 liming 密 码 为 aaa 
0 = 时 跳 转 到 welcome.jsp 页 
<jsp:forward page= "welcome.jsp"> 


<jsp:param name="fromPage" value="logincheck.jsp"/> 


</jsp:forward> 


} 
elset 


<jsp:forward page="error.jsp"/> 否则 跳 转 到 errorjsp 页 
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在 课堂 练习 4-6 的 基础 上 ,进一步 模拟 用 户 注册 的 功能 。 
要 求 : 当 用 户 单 击 注册 页 上 的 “注册 ”按钮 时 ,使 用 户 在 注册 页 上 输入 的 信息 能 提交 。 
提示 : 正确 选择 request 内 置 对 象 的 方法 来 完成 。 


闪 示 例 4-13 ”改写 后 的 注册 页 register. jsp 


<%e@page contentType="text/html;charset=GB18030" language="java" import= 
"java.util.x* "%> 
<HTML> 
<HEAD><TITLE> 注 册页 < /TITLE>< /HEAD> 
<BODY> 
<%@include file="menu.jsp"%> 
<FORM name= "forml" method= "post" action=""> 
<table width="316" border= "1" align= "center" bordercolor= "#666666" bgcolor= 
"CCCCCC"S 
尼克 
<td width="306" bgcolor= "#CCCC99" 
<table width="330" align="center" cellpadding="10"> 
<tr><td width= "73" height="30"> 用 户 名 </td> 
<td width="185"><input type="text" name= "username">< /td> 
< /tr> 
<tr> 
<td height="30"> 密 gnbsp;&nbsp; 码 </td> 
<td><input type="password" name="password"></td> 
</tr> 
到 二 工 福 
<td> 性 snbspyg&nbsp; 别 </td> 
<td>< input type="radio" name="gender" value: 


男 " /> 男 
<input type="radio" name="gender" value=" 女 " /> 女 </td> 
</tr> 
<tr> 
<td height="30"> 自 我 介绍 </td> 
<td><textarea name="introduction" rows="10" clos="36"></textarea>< /td> 
</tr> 
<tr> 
<td height="30"> 爱 snbsp;&nbsp; 好 </td> 
<td> 音 乐 <input type="checkbox" name="hobbycheckbox" value= "音乐"> 
绘画 <input type= "checkbox" name= "hobbycheckbox" value=" 绘 画 "></td> 
</tr> 
<tr> 
<td height="30"> 来 gnbsp;&nbsp; 自 </td> 
<td><select name= "comefrom"> 
<option> 罗 湖区 </option> 
<option> 福 田 区 </option> 
<option> 南 山区 </option> 
<option> 盐 田 区 </option> 
<option> 龙 岗 区 </option> 
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<option> 宝 安 区 </option> 
</select>< /td> 
</tr> 
<tr> 
<td colspan="2" align="center"><input type="submit" name="submit" value= 
"提交 ">< /ta> 
</tr> 
</table> 
</td> 
</tr> 
</table> 


</FORM> 
<% 

String username=request .getParameter ("username"); 

String password=request .getParameter ("password"); 

String gender=request .getParameter ("gender"); 

String introduction=request.getParameter ("introduction"); 

String comefrom=request .getParameter ("comefrom"); 

if(request.getParameterValues ("hobbycheckbox") !=null) 
String[] chl=request.getParameterValues ("hobbycheckbox"); 
for (int i=0;i<chl.length;i++){ 


out.println(chl[i]+"<br>"); 使 用 request 内 置 对 象 的 getParameterValues 
} 方法 接收 registerjsp 页 复 选 框 的 参数 


%> 
< /BODY> 
<HTML> 


一 练习 49 一 获取 客户 端的 JP 地址 - 
要 求 : 用 户 在 访问 网 站 时 ,在 导航 栏 中 显示 的 欢迎 信息 中 含有 用 户 的 IP 地 址 。 


效 示 例 4-14 ”改写 后 的 导航 栏 页 menu. jsp 


<%@page pageEncoding= "gb18030"%> 
<center> 你 好 来 自 <$=request .getRemoteAddr ()%> 的 朋友 < /center><br> 


使 用 request 内 置 对 象 的 
getRemoteAddr 方 法 


<table width="850" align="center" bgcolor==#ff6633> 获取 客户 端的 IP 地 址 
<th align="center"> 
<td align="left"> 
<a href="/4.2/1ogin.jsp"> 登 录 </a> 
<a href="/4.2/register.jsp"> 注 册 < /a> 
</th> 
</table> 
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4.5.2 response 对 象 


response 对 象 包含 了 响应 客户 请 求 的 有 关 信 息 , 它 将 浏览 器 参考 信息 ,如 回应 的 
heade、 网 页 采用 的 编码 集 等 提供 给 客户 端 ,但 在 JSP 中 很 少 直 接 用 到 它 。 它 是 
HttpServletResponse 方法 的 实例 。 

常用 方法 说 明 如 下 。 

(1) String getCharacterEncoding(): 返回 应 用 的 字符 编码 。 

(2) ServletOutputStream getOutputStream(): 返回 响应 的 一 个 二 进 制 输出 流 。 

(3) PrintWriter getWriter(): 返回 可 以 向 客户 端 输出 字符 的 一 个 对 象 。 

(4) void setContentLength(int len): 设置 响应 头 长 度 。 

(5) void setContentType(String type): 设置 响应 的 MIME 类 型 。 

(6) void sendRedirect(java. lang. String location) : 重新 定向 客户 端的 请 求 。 

response. sendRedirect() 与 一 jsp :forward 盖 的 区 别 如 下 。 
。 调用 sendRedirect( ) 方 法 的 重 定向 访问 过 程 结束 后 ,浏览 器 地 址 栏 中 显示 的 URL 
会 改变 ;而 使 用 forward 动作 不 会 改变 。 

。 sendRedirect 方法 响应 的 结果 就 是 告诉 浏览 器 去 重新 发 出 对 另外 一 个 URL 的 访 
问 请 求 (request)。 使 用 forward 动作 标签 跳 转 是 在 服务 器 端 内 部 将 请 求 转 发 给 
另外 一 个 资源 ,所 以 跳 转 前 后 属 同一 个 request 作用 域 。 


4.5.3 session 对象 


从 一 个 客户 打开 浏览 器 并 连接 到 服务 器 开始 ,到 客户 关闭 浏览 器 离开 这 个 服务 器 结 
东 , 被 称 为 一 个 会 话 (session)。 当 一 个 客户 访问 一 个 服务 器 时 ,可 能 会 在 这 个 服务 器 的 
几 个 页 面 之 间 反 复 连 接 、 反 复 刷 新 一 个 页 面 或 不 断 地 向 一 个 页 面 提交 信息 等 ,服务 器 应 当 
通过 某 种 办 法 知道 这 是 一 个 客户 ,这 就 需要 session 对 象 。 

session 对 象 用 来 保存 某 个 特定 客户 端 一 次 访问 的 一 些 特定 信息 。Session 对 象 在 第 
一 个 JSP 页 面 被 装载 时 自动 创建 ,完成 会 话 期 管理 。 当 一 个 客户 端 向 服务 器 发 送 第 一 个 
请 求 时 ,服务 器 为 其 建立 一 个 session ,并 为 此 session 创建 一 个 标识 号 。 这 个 用 户 随 后 的 
所 有 请 求 都 应 包括 这 个 标识 号 。 服 务 器 会 校对 这 个 标识 号 以 判断 请 求 属 于 哪个 session 。 
这 种 机 制 不 使 用 IP 作为 标识 ,是 因为 很 多 机 器 是 通过 代理 服务 器 方式 上 网 , 没 法 区 分 每 
一 台 机 器 。 当 用 户 关闭 浏览 器 并 释放 session ,或 者 服务 器 检测 到 session 非 活跃 状态 已 
经 超时 ,对 应 该 客户 端的 session 对 象 将 失效 ,客户 端 与 服务 器 的 会 话 关系 消失 。 

session 是 javax. Servlet. HttpSession 的 实例 。 

session 常用 方法 如 下 。 

(1) long getCreationTime() : 返回 session 创建 的 时 间 。 

(2) public String getId(): 返回 session 创建 时 JSP 引擎 为 它 设 置 的 唯一 ID 号 。 

(3) long getLastAccessedTime(): 返回 此 session 里 客户 端 最 近 一 次 请 求 的 时 间 。 
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(4) int getMaxJInactiveInterval() : 返回 两 次 请 求 间 隔 多 长 时 间 此 session 被 取消 。 

(5) String[] getValueNames(): 返回 一 个 包含 此 session 中 所 有 可 用 属性 的 数组 。 

(6) void invalidate(): 取消 session, 使 session 不 可 用 。 

(7) boolean isNew(): 返回 服务 器 创建 的 一 个 session ,确认 客户 端 是 否 已 经 加 入 。 

(8) void removeValue(String name) : 删除 session 中 指定 的 属性 。 

(9) void setMaxInactiveInterval (int seconds): 设置 两 次 请 求 间 隔 多 长 时 间 此 
session 会 被 取消 登录 。 


课堂 红 林 10 用 户 安 等 过 呵 
要 求 : 在 课堂 练习 4-8 的 基础 上 , 当 用 户 正确 登录 后 ,在 导航 栏 上 显示 欢迎 XXX 的 
信息 ;并 出 现 “ 退 出 ”功能 超 链接 , 当 单 击 “ 退 出 ”时 退出 追踪 并 显示 已 退出 的 提示 。 


闪 示 例 4-15 ”添加 了 记录 用 户 登 录 状 态 后 的 loginCheck. jsp 


<% 
String uname=request .getParameter ("username"); 
String pass=request .getParameter ("password"); 
if(uname!=null&g&pass!=null&&guname.equals ("liming") &&pass.equals ("aaa")) 


使 用 session 的 setAttribute 方 法 在 session 


session.setAttribute ("user",uname) :== 中 记录 用 户 的 登录 状态 。 在 session 存 储 
> 一 个 名 字 为 user 的 用 户 的 一 个 属性 


<jsp:forward page="welcome.jsp"> 
<jsp:param name="fromPage" value="logincheck.jsp"/> 
</jsp:forward> 


<jsp:forward page="error.jsp"/> 


多 > 
A 示例 4-16 ”判断 用 户 是 否 登录 menu. jsp 


<%@page pageEncoding= "gb2312"%> 
<center> 你 好 来 自 <%s= request .getRemoteAddr ()%> 的 朋友 < /center><br> 
<table width="850" align="center" bgcolor==#ff6633> 
<th align="center"> 
<td align="left"> 


<a href="/login.jsp"> 登 录 </a> 使 用 session 的 getAttribute 方 法 判断 用 户 
<a href="/register.jsp"> 注 册 < /a> 是 否 已 登录 ， 并 获取 已 登录 用 户 的 用 户 名 
<% 
ifl(session.getAttribute("acc") !=null) 
{ 
%> 
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<td align=right width=200> 用 户 <s= session.getRttribute ("acc")s> 已 登录 </td> 
<td align=center width=100><a href='logout.jsp'> 退 出 <td> 
<%} 
$$> 
</th> 
</table> 


已 登录 用 户 单 击 导 航 栏 中 的 “退出 ” 超 链接 ,将 实现 退出 登录 状态 的 功能 。 
闪 示 例 4-17 退出 登录 页 logout. jsp 


<%Q@page contentType="text/html; charset= gb2312" language="java" import= 
"java.util.* "%> 
<HTML> 

<HEAD><TITLE> 登 出 页 < /TITLE>< /HEAD> 

<BODY align="center"> 
< 


使 用 session 的 invalidate 方 法 使 session 
session.invalidate () /一 -一 失效 ， 从 而 达到 用 户 退 出 系统 的 日 的 


$%> 
<table width="360" align="center"> 
<tr><td align="center"> 欢 迎 下 次 再 来 !</td></tr> 
<tr><td align="center"><a href="login.jsp"> 返 回 登 录 页 </a></td></tr> 
</table> 
</BODY> 
<HTML> 


4.5.4 ”application 对象 


在 服务 器 启动 后 ,会 产生 一 个 application 对 象 。 当 一 个 客户 访问 服务 器 上 的 一 个 


JSP 页 面 时 ,JSP 引擎 为 客户 分 配 这 个 application 对 象 。 当 客户 在 所 访问 的 网 站 的 各 个 
页 面 之 间 浏 览 时 ,使 用 的 application 对 象 都 是 同一 个 ,直到 服务 器 关闭 ,这 个 application 
对 象 才 被 取消 。 


同一 


在 使 用 session 时 ,各 个 客户 独 享 各 自 的 session 对 象 ,而 使 用 application 时 ,客户 在 
个 服务 器 中 共享 一 个 application 对 象 , 因 此 .application 对 象 可 以 用 来 保存 服务 器 中 


一 些 公关 的 数据 。 
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application 是 javax. servlet. ServletContext 的 实例 。 

application 常用 方法 如 下 。 

(1) Object getAttribute(String name): 返回 给 定 对 象 的 属性 值 。 

(2) Enumeration getAttributeNames(): 返回 所 有 可 用 属性 名 的 枚 举 值 。 

(3) void setAttribute(String name,Object obj) : 设 定 对 象 的 属性 值 。 

(4) void removeAttribute(String name) : 删除 属性 及 属性 值 。 

(5) String getServerInfo() : 返回 JSP(SERVLET) 引 擎 名 及 版 本 号 。 

(6) String getRealPath(String path): 返回 一 虚拟 路 径 对 应 的 真实 路 径 。 

(7) ServletContext getContext (String uripath): 返回 指定 WebApplication 的 
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application 对 象 。 

(8) int getMajorVersion(): 返回 服务 器 支持 的 Servlet API 的 最 大 版 本 号 。 

(9) int getMinorVersion(): 返回 服务 器 支持 的 Servlet API 的 最 小 版 本 号 。 
(10) String getMimeType(String file) : 返回 指定 文件 的 MIME 类 型 。 
(11) URL getResource(String path) : 返回 指定 资源 (文件 及 目录 ) 的 URL 路 径 。 
(12) InputStream getResourceAsStream(String path): 返回 指定 资源 的 输入 流 。 
(13) RequestDispatcher getRequestDispatcher (String uripath): 返回 指定 资源 的 
RequestDispatcher 对 象 。 
(14) Servlet getServlet(String name) : 返回 指定 名 的 Servlet 。 
(15) Enumeration getServlets() : 返回 所 有 Servlet 的 枚 举 值 。 
(16) Enumeration getServletNames(): 返回 所 有 Servlet 名 的 枚 举 值 。 
(17) void log(String msg) : 把 指定 消息 写 入 Servlet 的 日 志文 件 。 
(18) void log(Exception exception,String msg): 把 指定 异常 的 栈 轨迹 及 错误 消息 
写 入 Servlet 的 日 志文 件 。 
(19) void log(String msg,Throwable throwable) : 把 栈 轨 迹 及 给 出 的 Throwable 异 
常 的 说 明 信 息 写 和 人 Servlet 的 日 志文 件 。 


课 熙 练 可 11 聊天 室 
要 求 : 在 课堂 练习 4-8 的 基础 上 , 当 用 户 正确 登录 后 ,在 导航 栏 上 显示 欢迎 X XX 的 信 
息 , 并 出 现 “ 退 出 ”功能 超 链接 , 当 单 击 该 超 链 接 时 退出 追踪 并 显示 已 退出 的 提示 。 


闪 示 例 4-18 ”聊天 室 chat. jsp 


<%@page contentType="text/html;charset=GB18030" language="java" import= 
"java.util.*%"%> 


<HTML> 
<HEAD> <TITLE> 聊 天 室 < /TITLE>< /HEAD> 
<BODY> 
<jsp:include page="menu.jsp"/> 
< 名 
if (session.getAttribute("acc")!=null) 
{ 
response.setHeader ("Refresh","15"); 
String userName= (String)session.getAttribute ("acc"); 
request .setCharacterEncoding ("GB2312"); 


String chat=request.getParameter ("mychat"); 
String chats= (String)application.getAttribute ("chat"); 


if(chat!=null) 
{ 
Date date=new Date(); 
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chat=userName+" ("+date.toString()+") "+Cchat7 
if(chats==nul1) 
chats=chat; 
else 
chats=chats+"\r\n"+chat; 
chat=null; 
} 
if (chats!=null) 
{ 
application.setAttribute ("chat",chats); 
out.print ("<table border=2 width= 400 bordercolorlight=#FFFFFF 
bordercolordark=#000000 cellspacing=0 cellpadding=0 align=center>"); 
out.print ("<tr>"); 
out .print ("<th> 聊 天 室 < /th>"); 
Out eprint ("</tr>")s 
out.print ("<tr>"); 
Out print(t"<ta>")? 
out.print ("<textarea name=introduction rows=10 clos=80 wrap=physical style 
=width:400>"+application.getAttribute ("chat")); 
out.print ("</textarea>"); 
out.print ("</td>"); 
out.print ("</tr>"); 
out.print ("</table>"); 
} 
out.print ("<table width= 400 cellspacing=0 cellpadding=0 align=center>"); 
out.print ("<tr height=30>"); 
out.print ("<th>"); 
out.print ("<Form action=chat.jsp method=post>"); 
out.print ("<input type=text size=30 name=mychat>"); 
out.print ("<input type=submit name=submit value= 发 言 >"); 
out.print ("</Form> "); 
out.print ("</th>"); 
out.print ("</tr>"); 
out.print ("</table>"); 
} 
else 
{ 
out.print ("<table width= 400 align= center>"); 
out.print ("<tr>"); 
out.print ("<th><a href='login.jsp'> 请 登录 < /a></th>"); 
out.print ("</tr>"); 
out.print ("<tr>"); 
out.print ("</table>"); 
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4.5.5 out 对 象 


out 对 象 主要 用 来 向 客户 端 输出 各 种 格式 的 数据 ,并 且 管 理应 用 服务 器 上 的 输出 组 
冲 区 。onut 是 javax. servlet. JspWriter 的 实例 。 

out 对 象 常用 方法 如 下 。 

(1) newLine(): 用 于 输出 一 个 换行 符 。 

(2) void flush(): 强制 输出 服务 器 中 的 数据 。 如 果 预 编译 指令 中 page 的 autoFlush 
属性 值 设 置 为 true, 那 么 JSP 程序 会 把 输出 数据 缓存 在 服务 器 的 缓冲 区 里 ,直到 程序 结束 
或 者 缓冲 区 充满 了 数据 ,服务 器 才 会 自动 把 缓冲 区 中 的 数据 输出 到 客户 端 。 如 果 在 JSP 
程序 里 使 用 了 flush( ) 方 法 ,那么 服务 器 不 管 缓冲 区 是 否 已 经 充满 值 ,都 将 数据 输出 到 客 
户 端 。 如 果 预 编译 指令 中 page 的 autoFlush 属性 值 设置 为 false, 那 么 需要 显 式 调用 
flush 将 数据 输出 到 客户 端 。 

(3) void close(): 该 方法 首先 将 缓冲 区 里 的 数据 输出 到 客户 端 , 然 后 关闭 对 客户 端 
的 输出 流 。 

(4) void clearBuffer() : 该 方法 用 于 清除 缓冲 区 里 的 数据 ,并 且 把 数据 写 到 客户 端 。 
在 缓冲 区 的 数据 为 空 的 时 候 , 这 个 方法 将 会 产生 IOException 错误 。 

(5) void clear(): 该 方法 用 于 清除 缓冲 区 里 的 数据 ,但 不 把 数据 写 到 客户 端 。 在 组 
冲 区 的 数据 为 空 的 时 候 , 这 个 方法 将 会 产生 IOException 错误 ,所 以 一 般 要 使 用 try…. 
catch... 块 包 住 。 

(6) int getBufferSize() : 该 方法 可 以 获取 缓冲 区 的 大 小 。 缓 冲 区 的 大 小 是 通过 预 纺 
译 指令 page 和 buffer 属性 来 确定 的 。 

(7) int getRemaining() : 该 方法 可 以 获得 缓冲 区 没有 使 用 的 字 节 数目 。 

(8) Boolean isAutoFlush(): 该 方法 返回 布尔 值 ,返回 值 由 page 指令 的 autoFlush 
属性 值 决定 。 


4.5.6 page 对 象 


page 对 象 就 是 指向 当前 JSP 页 面 本 身 , 它 是 java. lang. Object 的 实例 。 

pape 对 象 常 用 方法 如 下 。 

(1) class getClass: 返回 一 个 对 象 的 类 。 

(2) int hashCode(): 返回 一 个 对 象 的 hash 码 。 

(3) boolean equals(Object obj) : 判断 一 个 对 象 是 否 与 指定 的 其 他 对 象 相等 。 
(4) void copy(Object obj) : 把 一 个 对 象 复 制 到 指定 的 对 象 中 。 

(5) Object clone() : 克隆 一 个 对 象 。 

(6) String toString(): 把 一 个 对 象 转换 成 String 类 的 对 象 。 

(7) void notify() : 唤醒 一 个 等 待 的 线程 。 

(8) void notifyAll() : 唤醒 所 有 等 待 的 线程 。 
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(9) void wait(int timeout): 使 一 个 线程 处 于 等 待 直到 timeout 结束 或 被 唤醒 。 
(10) void wait() : 使 一 个 线程 处 于 等 待 ,直到 被 唤醒 。 

(11) void enterMonitor() : 对 一 个 对 象 加 锁 。 

(12) void exitMonitor() : 对 一 个 对 象 开 锁 。 


4.5.7 exception 对 象 


exception 对 象 是 一 个 例外 对 象 , 当 一 个 页 面 在 运行 过 程 中 发 生 了 例外 ,就 产生 了 这 
个 对 象 。 如 果 一 个 JSP 页 面 要 应 用 此 对 象 , 就 必须 把 isErrorPage 设 为 true, 和 否则 无 法 编 
译 程序 。exception 对 象 是 java. lang. Throwable 的 实例 。 

exception 对 象 常用 方法 如 下 。 

(1) String getMessage(): 返回 描述 异常 的 消息 。 

(2) String toString(): 返回 关于 异常 的 简短 描述 消息 。 

(3) void printStackTrace(): 显示 异常 及 其 栈 轨迹 。 

(4) Throwable FillInStackTrace(): 重 写 异常 的 执行 栈 轨迹 。 


4.5.8 pageContext 对 象 


pageContext 对 象 提 供 了 对 JSP 页 面 内 所 有 对 象 及 名 字 空 间 的 访问 ,可 以 访问 本 页 
所 在 的 session, 也 可 以 读 取 本 页 面 所 在 的 application 对 象 的 某 一 属性 值 ,相当 于 页 面 中 
所 有 功能 的 集大成 者 , 它 是 javax. servlet. jsp. PageContext 的 实例 。 

pageContext 对 象 常 用 方法 如 下 。 

(1) JspWriter getOut(): 返回 当前 客户 端 响应 被 使 用 的 JspWriter 流 (out)。 

(2) HttpSession getSession(): 返回 当前 页 中 的 HttpSession 对 象 (session ) 。 

(3) Object getPage() : 返回 当前 页 的 Object 对 象 (page)。 

(4) ServletRequest getRequest() : 返回 当前 页 的 ServletRequest 对 象 (request) 。 

(5) ServletResponse getResponse ( ): 返回 当前 页 的 ServletResponse 对 象 
(response) 。 

(6) Exception getException(): 返回 当前 页 的 Exception 对 象 Cexception) 。 

(7) ServletConfig getServletConfig(): 返回 当前 页 的 ServletConfig 对 象 (config)。 

(8) ServletContext getServletContext( ): 返回 当前 页 的 ServletContext 对 象 
(application) 。 

(9) void setAttribute(String name,Object attribute): 设置 属性 及 属性 值 。 

(10) void setAttribute(String name,Object obj,int scope) : 在 指定 范围 内 设置 属性 
及 属性 值 。 

(11) public Object getAttribute(String name) : 取 属 性 的 值 。 

(12) Object getAttribute(String name,int scope): 在 指定 范围 内 取 属 性 的 值 。 

(13) public Object findAttribute (String name): 寻找 一 属性 ,返回 其 属性 值 或 
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(14) void removeAttribute(String name): 删除 某 属 性 。 

(15) void removeAttribute(String name,int scope) : 在 指定 范围 内 删除 某 属性 。 

(16) int getAttributeScope(String name): 返回 某 属性 的 作用 范围 。 

(17) Enumeration getAttributeNamesInScope(int scope): 返回 指定 范围 内 可 用 的 
属性 名 枚 举 。 

(18) void release(): 释放 pageContext 所 占用 的 资源 。 

(19) void forward(String relativeUrlPath): 使 当前 页 面 重 导 到 另 一 页 面 。 

(20) void include(String relativeUrlPath) : 在 当前 位 置 包含 另 一 文件 。 


4.5.9 config 对 象 


config 对 象 是 javax. servlet. ServletConfig 的 实例 ,是 在 一 个 Servlet 初始 化 时 ,JSP 
引擎 向 它 传递 信息 用 的 ,此 信息 包括 Servlet 初始 化 时 所 要 用 到 的 参数 (通过 属性 名 和 属 
性 值 构成 ) 以 及 服务 器 的 有 关 信 息 ( 通 过 传递 一 个 ServletContext 对 象 ) 。 

config 对 象 常用 方法 如 下 。 

(1) ServletContext getServletContext(): 返回 含有 服务 器 相关 信息 的 ServletContext 
对 象 。 

(2) String getInitParameter(String name) : 返回 初始 化 参数 的 值 。 

(3) Enumeration getInitParameterNames(): 返回 Servlet 初始 化 所 需 所 有 参数 的 
枚 举 。 


4.6 JDBC 简 介 


4.6.1 JDBC 的 概念 及 特点 


JDBC(Java Database Connectivity,Java 数据 库 连 接 ) 是 一 种 用 于 执行 SQL 语句 的 
Java API, 可 以 为 多 种 关系 数据 库 提 供 统一 的 访问 接口 。JDBC 由 一 组 用 Java 语言 编写 
的 类 与 接口 组 成 ,通过 调用 这 些 类 和 接口 所 提供 的 方法 ,用 户 能 够 以 一 致 的 方式 连接 多 种 
不 同 的 数据 库 系统 (如 Access Server 2000、Oracle、Sybase 等 ) ,进而 使 用 标准 的 SQL 语 
言 来 存 取 数据 库 中 的 数据 ,而 不 必 再 为 每 一 种 数据 库 系统 编写 不 同 的 Java 程序 代码 。 

简单 地 说 ,JDBC 能 完成 下 列 三 个 功能 : 

。 与 一 个 数据 库 建立 连接 ; 

。 向 数据 库 发 送 SQL 指令 ; 
。 处 理 数 据 库 返 回 的 结果 。 
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4.6.2 Web 访问 数据 库 的 原理 


Web 访问 数据 库 的 一 般 流程 如 图 4-14 所 示 , 客 户 端的 请 求 以 JSP 或 者 HTML 的 形 
式 发 送 到 服务 器 ,经 Web 服务 器 解析 后 ,对 于 需要 进行 数据 库 访 问 的 业务 逻辑 , Web 服 
务 器 将 通过 JDBD 的 形式 连接 数据 库 服 务 器 (如 Oracle、SQL Server、MySQL 等 ) ,数据 库 
服务 器 执行 数据 表 中 记录 的 增 、 删 、 改 、 查 等 操作 ,并 将 结果 返回 给 Web 服务 器 , Web 服 
务 器 再 将 操作 数据 库 的 结果 以 HTML 的 形式 通过 HTTP 协议 回 传 给 前 端的 浏览 器 。 


客户 站 及 务 器 端 
JSPIHTML | | Ws。|_JDBC _| 数据 
浏览 器 口 服务 器 | | 数据 库 


图 4-14 Web 操作 数据 库 流程 图 


4.6.3 ” JDBC 的 结构 


JDBC API 通过 一 个 数据 库 管理 器 (Database Manager) 和 为 各 种 数据 库 定制 的 驱动 
程序 提供 与 不 同 数据 库 的 透明 连接 。JDBC 数据 库 管理 器 将 确保 正确 的 驱动 程序 被 用 于 
连接 数据 源 。 它 可 以 同时 支持 与 不 同 数据 库 的 连接 。 

JDBC 数据 库 管理 器 将 标准 的 JDBC 指令 转换 成 适用 于 不 同 数据 库 通信 的 网 络 协议 
指令 或 其 他 API 指令 。 这 种 指令 的 转换 机 制 , 使 基于 JDBC 接口 开发 的 程序 可 以 独立 于 
数据 库 的 种 类 。 如 果 底 层 的 数据 库 被 更 换 了 ,用 户 只 需 相 应 的 替换 程序 中 所 引用 的 
JDBC 驱动 程序 即 可 。 


4.6.4 ”JDBC 的 种 类 


数据 库 连 接 对 动态 网 站 来 说 是 最 重要 的 部 分 。 很 多 数据 库 系 统 带 有 JDBC 驱动 程 
序 , 通 过 JDBC 驱动 程序 与 数据 库 相 连 ,可 以 执行 查询 和 提取 数据 等 操作 。 目 前 ,JDBC 的 
驱动 程序 主要 有 以 下 四 种 。 


1. JDBC-ODBC 桥 


驱动 程序 用 于 连接 JDBC 和 另 一 种 数据 库 连 接 机 制 ODBC,JDBC-ODBC 桥 使 基于 
JDBC 的 程序 能 通过 传统 的 ODBC 驱动 程序 访问 数据 库 。 由 于 大 多 数 数据 库 系 统 都 带 有 
ODBC 驱动 程序 ,所 以 Java 程序 能 访问 诸如 Oracle、Sybase、MS SQL Server 和 MS 
Access 等 数据 库 。 这 一 驱动 程序 已 被 包括 在 Java 2 SDK 的 sun. jdbc. odbc 包 中 。 这 种 
方法 要 求 客户 端 必须 安装 ODBC 驱动 ,所 以 不 适合 Web 应 用 。 
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2. 本 地 API 驱动 


本 地 API 驱动 直接 把 JDBC 调用 转变 为 数据 库 的 标准 调用 再 去 访问 数据 库 。 这 种 
驱动 比 JDBC-ODBC 桥 执行 效率 高 .但 是 ,仍然 需要 在 客户 端 加 载 数 据 库 厂商 提供 的 类 
库 , 因 此 ,同样 不 适合 Web 应 用 。 


3. Java 到 网 络 协议 


这 类 驱动 程序 通过 一 定 的 网 络 协议 与 数据 库 服务 器 上 的 JDBC 中 间 件 (Middleware) 
通信 。 这 一 中 间 件 程序 将 网 络 协议 指令 转换 成 数据 库 指令 。 此 类 驱动 程序 不 需要 在 客户 
端 安装 目标 数据 库 的 类 库 。 


4. Java 到 数据 库 协议 


此 类 了 驱动 通过 实现 一 定 的 数据 库 协议 直接 与 数据 库 通信 。 与 Java 到 网 络 协 议 的 驱 
动 程序 一 样 , 它 直 接 与 数据 库 通信 ,所 以 其 效率 是 4 种 驱动 程序 中 最 高 的 。 其 缺点 是 : 当 
目标 数据 库 的 类 型 更 换 时 ,必须 重新 购买 JDBC 驱动 程序 并 在 本 地 安装 。 

究竟 选择 哪 一 类 型 的 驱动 程序 ,完全 取决 于 所 构建 Web 应 用 的 需求 。 如 果 不 希 望 在 
程序 客户 端 安装 大 量 目标 数据 库 的 类 库 , 则 应 该 考虑 后 两 种 驱动 程序 ;如 果 后 两 种 驱动 程 
序 的 价格 特别 高 ,那么 可 以 使 用 前 两 种 驱动 程序 。 


4.6.5 手动 建立 ODBC 数据 源 


下 面 以 一 个 实例 来 说 明 如 何 使 用 JDBC 连接 数据 库 ,实例 中 为 便于 操作 ,采用 了 
JDBC-ODBC 桥 的 方式 连接 Access 数据 库 。 


课堂 练习 三 12”JSP 读 取 数据 库 的 信息 

要 求 : 创建 一 个 Access 数据 库 db. mdb, 在 数据 库 中 创建 一 个 数据 表 student, 包含 
的 字段 如 表 4-4 所 示 。 编 写 JSP 文件 ,查询 数据 表 中 所 有 “软件 学 院 ” 的 学 生 并 显示 在 浏 
览 器 中 。 


表 4-4 数据 表 中 的 字段 


说 明 


id 序号 ,主键 
name 姓名 
school 所 属 学 院 


年 龄 


age 


采用 这 种 方式 连接 数据 库 ,需要 在 系统 中 设置 ODBC 数据 源 ,在 Windows 系统 中 设 
置 数据 源 的 步 又 如 下 。 
(1) 进入 ODBC 数据 源 环境 
打开 Windows 中 的 "控制 面板 ”, 找 到 ”管理 工具 ”中 的 "数据 源 (ODBC) "工具 图 标 ， 
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如 图 4-15 所 示 。 国王 

双击 “数据 源 (ODBC)”, 打 开 “ODBC 数据 源 管理 GEARG 和 
器 ”对话 框 ,选择 “系统 DSN” 选 项 卡 , 单 击 右 侧 的 “ 添 ”|。 罩 字 ” 回 扔 f 巴 电 了 闻 件 四 刘 
加 ”按钮 ,如 图 4-16 所 示 。 

提示 : 在 这 里 也 可 以 选择 用户 DSN ”选项 卡 。 
“系统 DSN” 和 “用 户 DSN” 在 使 用 上 的 区 别 是 :“ 用 户 
DSN” 上 设置 的 DSN 只 能 在 使 用 进行 设置 操作 的 用 户 


图 4-15 设置 数据 源 


(1) 选择 “系统 DSN" 选 项 卡 


Microsoft Access Driver (rmdb) 人 〇 ) 单 击 " 添 加 按钮 
Wierosoft Access Driver 人 mdb) 

LiBDataBase ~ Microsoft Access Driver (#.mdb) 

MySiteDatabase Microsoft Access Driver (#.mdb) 

PinCan Microsoft Access Driver (x.ndb) 

TiJian Microsoft Access Driver (#.ndb) 
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图 4-16 ODBC 数据 源 管理 器 


名 登录 时 才 可 以 使 用 ;而 "系统 DSN” 则 针对 任何 登录 的 用 户 都 可 以 使 用 。 

(2) 选择 数据 源 的 驱动 程序 类 型 

在 出 现 的 “创建 新 数据 源 ” 对 话 框 中 选择 Access 数据 库 的 驱动 程序 Microsoft 
Access Driver( * . mdb)”, 青 单 击 “ 完 成 ”按钮 ,如 图 4-17 所 示 。 


Driver da Microsoft para arquivos texto 体 | 

Driver do Microsoft Access (*.mdb) 

Driver do Microsoft dBase (*. dbf) 

Driver do Microsoft Excel (#.xls) 

Driver do Microsoft Paradox (*.db ) (1) 选中 Access 数 据 


Driver para o Microsoft Visual FoxFro a 
Microsoft Access dBASE Driver (*. db fi 库 的 驱动 程序 


2) HE | 


图 4-17 ODBC 数据 源 管理 器 
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(3) 为 新 数据 源 命名 ,并 与 指定 的 数据 库 关联 


在 出 现 的 "ODBC Microsoft Access 安装 ”对 话 框 中 输入 数据 源 的 名 称 , 然 后 单 击 " 选 
择 ” 按 钮 ,如 图 4-18 所 示 。 


(D 输入 数据 源 的 名 称 


(2) 单 击 “ 选 择 " 按 钮 


系统 示 据 库 Y) ] 


选 页 四 > 
图 4-18 输入 数据 源 名 称 


在 出 现 的 "选择 数据 库 ” 对 话 框 中 ,选择 数据 库 所 在 的 文件 夹 后 ,数据库 会 自动 出 现在 


左 侧 列表 框 中 ,再 单 击 数 据 库 名 称 ,可 以 使 之 出 现在 上 方 栏 中 ,然后 单 击 “ 确 定 ” 按 钮 ,如 
图 4-19 所 示 。 


数据 库 名 目录 名); 
dN \webapps\root 
(3) 选择 与 数据 源 -一 
关联 的 数据 库 Promrm Files 
Toncat 7.0 


它 vebapps 
[2 


文件 类 型 0T): 驱动 器 o) 
[eeess 数据 库 ce wzj [a 


图 4-19 指定 数据 库 


现在 返回 到 *ODBC Microsoft Access 安装 "对 话 框 ,这 时 可 以 看 到 数据 源 的 名 称 和 
与 之 关联 的 数据 库 , 单 击 “确定 "按钮 ,如 图 4-20 所 示 。 


0 Bn 确定 
说 明 中 ); -一 一 
本 | 


数据 库 : Di\...\Toncat T.Ovvebapps\EDOT\ab mab 必 助 00 
ET | Ee)... | 修复 四 .| 压缩 各. a 
三 系统 数据 库 
6 无 中 
个 数据 库 全 ); 
| 3] 


图 4-20 完成 数据 源 名 称 与 数据 库 的 关联 
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接着 返回 到 “ODBC 数据 源 管理 器 ?对 话 框 ,这 时 可 以 看 到 新 设置 的 数据 源 的 名 称 已 
经 出 现在 数据 源 列表 中 , 单 击 “ 确 定 ” 按 钮 ,完成 数据 源 的 设置 ,如 图 4-21 所 示 。 


用 户 DSH 系统 DSH | 文件 Ds | 驱动 程序 | 跟踪 ”| 连接 地 | 关于 | 
系统 郝 据 大 全); 


Microsoft Access Driver (#.mdb) 
Microsoft Access Driver (#.mdb) 
ierese 

单 击 " 配 置 ” 
i 按钮 可 以 修 
icrese ee iver (tm 


确定 取消 应 用 人) 帮助 


图 4-21 完成 数据 源 的 设置 


如 果 是 在 用 户 DSN 选项 卡 中 进行 以 上 设置 ,还 应 继续 设置 用 户 的 登录 名 称 和 密码 ， 
方法 如 下 。 


在 建立 了 数据 源 名 称 与 数据 库 的 关联 后 ,还 要 单 击 "ODBC Microsoft Access 安装 ” 
对 话 框 中 的 “高 级 ”按钮 ,为 数据 源 设置 一 个 登录 名 称 和 密码 ,如 图 4-22 和 图 4-23 所 示 


数据 大 名 0); time | 
说 明 @): 
pe -一 
数据 库 : 0D: 和 .Tomeat 7.0\webapps\ROOT\db. mdb 帮助 00 
创 旱 0. . | 个 夏 中. | _ 压 第 wm | i 
系统 数据 库 
G 无 @) 
个 数据 库 上 T)- 

系统 数据 库 Y) | i 


图 4-22 设置 数据 源 的 高 级 属性 


设置 好 Access 数据 库 的 数据 源 之 后 ,这 个 数据 源 Student 就 对 应 着 一 个 Access 数据 


库 , 即 student. mdb。 用 户 在 编写 Java 程序 时 ,就 可 以 将 这 个 数据 源 作为 数据 库 的 连接 对 
象 包含 在 源 程序 中 ,以 便 对 存储 在 数据 库 中 的 数据 进行 存 取 操作 。 


说 明 : 对 于 不 同类 型 的 数据 库 , 在 建立 ODBC 数据 源 时 ,应 选择 相应 的 数据 库 驱 动 
程序 。 
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(D 输入 登录 名 称 


(2) 输入 密码 


图 4-23 设置 登录 名 称 与 密码 


4.6.6 JDBC 访问 数据 库 的 基本 步骤 


JDBC 提供 了 Java 应 用 程序 与 各 种 不 同 数据 库 之 间 进 行 对 话 的 接口 。 

使 用 JDBC 访问 数据 库 的 步骤 如 下 : 

第 一 步 ” 引 入 java sql 包 

<%@page import="java.sql.* "和 > 

第 二 步 ”加载 JDBC 驱动 程序 

在 与 某 一 特定 数据 库 建立 连接 之 前 ,必须 首先 加 载 一 种 可 用 的 JDBC 驱动 程序 。 这 
需要 使 用 下 列 方法 来 加 载 JDBC 驱动 程序 : 

Class.forName ("DriverName"); 


使 用 Class 类 的 forName() 方 法 来 装载 数据 库 驱 动 类 ,并 且 进 行 类 的 初始 化 等 操作 。 
DriverName 是 要 加 载 的 JDBC 驱动 程序 名 称 。 驱 动 程序 名 称 根据 数据 库 厂商 提供 的 
JDBC 驱动 程序 的 种 类 来 确定 。 对 于 JDBC-ODBC 桥 , 加 载 JDBC-ODBC 数据 库 驱 动 程序 
的 方法 为 ， 


Class .forName ("sun.jdbc.odbc.JdbcOdbcDriver"); 

第 三 步 ” 创 建 数 据 库 连接 

在 使 用 JDBC 操作 数据 库 之 前 ,必须 首先 建立 一 个 数据 库 连接 ,创建 和 指定 数据 库 的 
连接 需要 使 用 DriverManager 类 的 getConnection() 方 法 ,其 一 般 的 使 用 格式 如 下 : 


Connection conn = DriverManager. getConnection (String url, String user, String 
password); 


该 方法 返回 的 是 一 个 java. sql. Connection 对 象 。 这 里 的 URL 是 一 个 字符 串 ,代表 
了 将 要 连接 的 数据 源 , 即 具体 的 数据 库 位 置 。 不 同 的 JDBC 驱动 程序 其 URL 格式 是 不 同 
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的 。 通 过 设置 数据 源 的 方式 连接 4. 6. 6 小 节 中 student. mdb 数据 库 的 代码 示例 如 下 : 


Class.forName ("sun.jdbc.ordc.JdbcOdbcDriver"); 


String user=""; - 
String password=""; 数据 源 名 称 


Connection Conn = DriverManager. getConnection ( " jdbc: odbc: Student ", user, 


password); 


第 四 步 ” 创 建 Statement 或 者 预 编 译 Statement 对 象 

在 与 某 个 特定 数据 库 建立 连接 之 后 ,这 个 连接 会 话 就 可 以 用 于 发 送 SQL 语句 。 在 发 
送 SQL 语句 之 前 ,必须 创建 一 个 Statement 类 的 对 象 ,该 对 象 负责 将 SQL 语句 发 送 给 数 
据 库 。 如 果 SQL 语句 运行 后 产生 结果 集 , Statement 对 象 会 将 结果 集 返 回 给 一 个 
ResultSet 对 象 。 

(1) 创建 Statement 对 象 的 方法 

创建 Statement 对 象 是 使 用 Connection 接口 的 createStatement() 方 法 来 实现 的 : 


Statement smt= conn .CreateStatement () 7 


还 可 以 使 用 预 编译 statement 的 方式 来 向 数据 库 发 送 SQL 语句 和 接受 结果 集 。 

预 编译 语句 PreparedStatement 是 java. sql 中 的 一 个 接口 , 它 是 Statement 的 子 接 
口 。 通 过 Statement 对 象 执行 SQL 语句 时 ,需要 将 SQL 语句 发 送 给 数据 库 管理 系统 
(DataBase Management System, DBMS) ,由 DBMS 进行 编译 后 再 执行 。 预 编译 语句 和 
Statement 不 同 ,在 创建 PreparedStatement 对 象 时 就 指定 了 SQL 语句 ,该 语句 立即 发 送 
给 DBMS 进行 编译 。 当 该 编译 语句 被 执行 时 ,DBMS 直接 运行 编译 后 的 SQL 语句 。 

(2) 预 编译 SQL 语句 的 方法 

Oz@ 创建 PreparedStatement 对 象 


PreparedStatement psmt= conn.prepareStatement (" SELECT * FROM employee WHERE 
name= ? AND password=?"); 


创建 包含 带 两 个 IN 参数 占 位 符 “?” 的 SQL 语句 的 PreparedStatement 对 象 。 

@ 传递 IN 参数 

在 执行 PreparedStatement 对 象 之 前 ,必须 设置 每 个 “?” 参 数 的 值 。 这 可 通过 调用 
setXXX() 方 法 来 完成 ,其 中 XXX 是 与 该 参数 相应 的 类 型 。 以 下 代码 将 第 一 个 参数 设 为 
"liming" ,第 二 个 参数 设 为 "123456": 


pstmt .setString (1, "liming"); 
pstmt .setstring(2, "123456"); 


一 旦 设置 了 给 定语 句 的 参数 值 , 其 值 将 一 直 保 留 , 直到 被 设置 为 新 值 或 者 调用 
clearParameters() 方 法 清除 它 为 止 。 

(3) 使 用 预 编译 语句 的 优点 

J@ 提高 效率 。 当 需要 对 数据 库 进行 数据 插入 、 更 新 或 者 删除 的 时 候 , 程 序 会 发 送 整 
个 SQL 语句 给 数据 库 处 理 和 执行 。 数 据 库 处 理 一 个 SQL 语句 ,需要 完成 解析 SQL 语 
句 、 检 查 语法 和 语义 以 及 生成 代码 ;一般 说 来 ,处 理 时 间 要 比 执行 语句 所 需要 的 时 间 长 。 
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预 编译 语句 在 创建 的 时 候 已 经 是 将 指定 的 SQL 语句 发 送 给 了 DBMS ,完成 了 解析 、 检 查 、 
编译 等 工作 。 因 此 , 当 一 个 SQL 语句 需要 执行 多 次 时 ,使 用 预 编译 语句 可 以 减少 处 理 时 
间 ,提高 执行 效率 。 

@ 提高 安全 性 。 可 以 防止 恶意 的 SQL 语句 注入 。 


String sql="SELECT * FROM employee WHERE name= ' "+username+"' AND password= 
'"+password+" '" 


如 果 把 "' "作为 username 的 参数 以 及 "'OR true" 作 为 password 传人 进来 ,SELECT 
语句 就 变 成 : 


SELECT * FROM employee WHERE name= '' RND password='' OR true 


条 件 部 分 的 布尔 运算 结果 为 true, 所 以 任意 的 用 户 名 和 密码 都 可 以 通过 验证 。 而 如 
果 使 用 预 编译 语句 ,传人 的 任何 参数 内 容 不 会 和 原来 的 语句 发 生 任何 匹配 的 关系 ,所 以 使 
用 预 编译 可 以 防止 恶意 的 SQL 注入 。 

当 语 句 格 式 固定 时 ,应 使 用 PreparedStatement; 当 语 句 格 式 无 法 预见 时 , 才 考 虑 采用 
Statement 。 

第 五 步 ” 执 行 SQL 语句 

Statement 对 象 创建 好 之 后 ,就 可 以 使 用 该 对 象 的 executeQuery( ) 方 法 来 执行 数据 
库 查 询 语句 。executeQuery() 方 法 返回 一 个 ResultSet 类 的 对 象 , 它 包含 了 SQL 查询 语 
句 执行 的 结果 。 例 如 : 


ResultSet rs=smt .executeQuery ("SELECT * from student WHERE school=' 软 件 学 院 '"); 


当 使 用 预 编译 PreparedStatement 的 方式 时 ,仍然 使 用 该 对 象 的 executeQuery( ) 方 
法 来 执行 数据 库 查询 语句 ,但 由 于 SQL 语句 在 前 面 的 步骤 中 已 经 “准备 好 了 ”所 以 
PreparedStatement 的 executeQuery() 方 法 不 需要 SQL 语句 作为 参数 ,执行 的 结果 仍然 
是 一 个 包含 了 SQL 查询 语句 执行 结果 ResultSet 类 的 对 象 , 示 例如 下 : 


ResultSet rs=psmt .executeQuery (); 


第 六 步 ”接收 并 处 理 SQL 的 返回 结果 

JDBC 接收 结果 是 通过 ResultSet 类 的 对 象 来 实现 的 。 一 个 ResultSet 对 象 包含 了 执 
行 某 个 SQL 语句 后 满足 条 件 的 所 有 的 行 , 它 还 提供 了 对 这 些 行 的 访问 ,用 户 可 以 通过 一 
组 getters 方法 来 访问 当前 行 的 不 同 列 。 通 常 结果 集 的 形式 是 一 张 带 有 表 头 和 相应 数据 
的 表 。 

(1) ResultSet 的 getters 方法 

@ BigDecimal getBigDecimal (int columnIndex): 以 具有 全 部 精度 的 java. math. 
BigDecimal 对 象形 式 获 取 当 前 行 中 某 个 列 的 值 。Palm OS 的 DB2 Everyplace JDBC 驱动 
程序 不 支持 此 方法 。 

@ BigDecimal getBigDecimal(int columnIndex , int scale): 以 Java 编程 语言 中 的 
java. math. BigDecimal 对 象形 式 获 取 此 ResultSet 对 象 当 前 行 中 指定 列 的 值 。Palm OS 
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的 DB2 Everyplace JDBC 驱动 程序 不 支持 此 方法 。 

@ BigDecimal getBigDecimal (String columnName)JDBC 2.0: 以 具有 全 部 精度 的 
java. math. BigDecimal 对 象形 式 获取 当前 行 中 某 个 列 的 值 。Palm OS 的 DB2 Everyplace 
JDBC 驱动 程序 不 支持 此 方法 。 

@ BigDecimal getBigDecimal(String columnName, int scale): 以 Java 编程 语言 中 
的 java. math. BigDecimal 对 象形 式 获取 此 ResultSet 对 象 当 前 行 中 指定 列 的 值 。Palm 
OS 的 DB2 Everyplace JDBC 驱动 程序 不 支持 此 方法 。 

@ Blob getBlob (int columnIndex) : 获取 此 ResultSet 对 象 的 当前 行 中 的 BLOB 值 。 

@ Blob getBlob(String columnName) : 获取 此 ResultSet 对 象 的 当前 行 中 的 BLOB 值 。 

@ boolean getBoolean (int columnIndex) : 以 Java 布尔 值 形 式 获 取 当 前 行 中 指定 列 
的 值 。 

@@ boolean getBoolean (String columnName): 以 Java 布尔 值 形 式 获 取 当 前 行 中 指 
定名 称 列 的 值 。 

@ byte getByte (int columnIndex): 以 Java 编程 语言 中 的 字 节 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 序号 列 的 值 。 

加 byte getByte (String columnName): 以 Java 编程 语言 中 的 字 节 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 名 称 列 的 值 。 

@ byte[] getBytes(int columnIndex) : 以 Java 编程 语言 中 的 字 节 数组 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 序号 列 的 值 。 

@ byte[] getBytes (String columnName) : 以 Java 编程 语言 中 的 字 节 数组 形式 获取 
此 ResultSet 对 象 当 前 行 中 指定 名 称 列 的 值 。 

@ int getConcurrency(): JDBC 2.0: 返回 结果 集 的 并 行 性 方式 。 

蝎 Date getDate (int columnIndex) : 以 Java 编程 语言 中 的 java. sql. Date 对 象形 式 
获取 此 ResultSet 对 象 当 前 行 中 指定 序号 列 的 值 。 

a Date getDate(int columnIndex,Calendar cal) : 以 Java 编程 语言 中 的 java. sql. 
Date 对 象形 式 返 回 此 ResultSet 对 象 的 当前 行 中 指定 序号 列 的 值 。 

@@ Date getDate(String columnName) : 以 Java 编程 语言 中 的 java. sql. Date 对 象形 
式 获取 此 ResultSet 对 象 的 当前 行 中 指定 名 称 列 的 值 。 

@ double getDouble(int columnIndex) : 以 Java 双 精 度 形式 获取 当前 行 中 指定 序号 
列 的 值 。 

@@ double getDouble(String columnName): 以 Java 双 精 度 形式 获取 当前 行 中 指定 
名 称 列 的 值 。 

四 float getFloat(int columnIndex): 以 Java 浮 点 形式 获取 当前 行 中 某 序号 列 的 值 。 

加 float getFloat(String columnName) : 以 Java 浮 点 形式 获取 当前 行 中 指定 名 称 列 
的 值 。 

加 int getInt (int columnIndex) : 以 Java 编程 语言 中 的 整数 形式 获取 此 ResultSet 
对 象 当 前 行 中 指定 序号 列 的 值 。 
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四 int getInt (String columnName): 以 Java 编程 语言 中 的 整数 形式 获取 此 
ResultSet 对 象 的 当前 行 中 指定 名 称 列 的 值 。 

四 long getLong(int columnIndex) : 以 Java 长 整 型 形式 获取 当前 行 中 指定 序号 列 
的 值 。 

四 long getLong(String columnName) : 以 Java 长 整 型 形式 获取 当前 行 中 指定 名 称 
列 的 值 。 

四 ResultSetMetaData getMetaData() : 检索 此 ResultSet 对 象 的 列 的 数目 .类 型 和 
属性 。 

四 Object getObject(int columnIndex): 以 Java 对 象形 式 获取 当前 行 中 指定 序号 列 
的 值 。 

四 Object getObject(String columnName): 以 Java 对 象形 式 获取 当前 行 中 指定 名 称 
列 的 值 。 

四 int getRow()JDBC 2.0: 检索 当前 行 号 。 

四 short getShort (int columnIndex): 以 Java 编程 语言 中 的 short 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 序号 列 的 值 。 

0 short getShort(String columnName): 以 Java 编程 语言 中 的 short 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 名 称 列 的 值 。 

团 Statement getStatement()JDBC 2.0: 返回 产生 此 ResultSet 对 象 的 “语句 ”。 

曙 String getString(int columnIndex): 以 Java 编程 语言 中 的 String 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 序号 列 的 值 。 

图 String getString(String columnName) : 以 Java 编程 语言 中 的 String 形式 获取 此 
ResultSet 对 象 当 前 行 中 指定 名 称 列 的 值 。 

名 Time getTime(int columnIndex) : 以 Java 编程 语言 中 的 java. sql. Time 对 象形 式 
获取 此 ResultSet 对 象 的 当前 行 中 指定 序号 列 的 值 。 

国 Time getTime(String columnName): 以 Java 编程 语言 中 的 java. sql. Date 对 象 
形式 获取 此 ResultSet 对 象 的 当前 行 中 指定 名 称 列 的 值 。 

8 Timestamp getTimestamp (String columnName): 以 Java 编程 语言 中 的 java. 
sql. Timestamp 对 象形 式 获取 此 ResultSet 对 象 的 当前 行 中 指定 名 称 列 的 值 。 

多 Timestamp getTimestamp (int columnIndex): 以 Java 编程 语言 中 的 java. sql. 
Timestamp 对 象形 式 获 取 此 ResultSet 对 象 的 当前 行 中 指定 序号 列 的 值 。 

(2) 与 游标 相关 的 方法 

@ boolean isAfterLast() : 指示 游标 是 否 在 结果 集中 的 最 后 一 行 后 面 。 

@ boolean isBeforeFirst() : 指示 游标 是 否 在 结果 集中 的 第 一 行 前 面 。 

@ boolean isFirst() : 指示 游标 是 否 在 结果 集中 的 第 一 行 上 。 

@ boolean isLast( ): 指示 游标 是 否 在 结果 集中 的 最 后 一 行 上 。 对 于 具有 类 型 
TYPE_FORWARD_ONLY 的 结果 集 , 不 支持 此 方法 。 

@ boolean last() : 将 游标 移 至 结果 集中 的 最 后 一 行 。 
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@ boolean next() : 将 游标 从 当前 位 置 向 下 移动 一 行 。 

@ boolean previous() : 将 游标 移 至 结果 集中 的 前 一 行 。 

图 boolean relative(int rows) : 将 游标 移动 相对 数量 的 行 数 ,可 以 是 正 数 或 负数 。 
@ boolean absolute(int row) : 将 游标 移 至 结果 集中 的 给 定 行 号 。 

@ void afterLast(): 将 游标 移 至 结果 集 的 末尾 ,正好 在 最 后 一 行 的 后 面 。 

加 void beforeFirst(): 将 游标 移 至 结果 集 的 前 方 ,正好 在 第 一 行 的 前 面 。 

加 boolean first(): 将 游标 移 至 结果 集中 的 第 一 行 。 

(3) 其 他 方法 

@ boolean wasNull() : 报告 读 取 的 最 后 一 列 是 否 具 有 值 SQL NULL。 

@ int getType()JDBC 2.0: 返回 此 结果 集 的 类 型 。 

@ SQLWarning getWarnings(): 返回 此 ResultSet 上 的 调用 报告 的 首次 警告 。 
@ void clearWarnings(): 清除 此 ResultSet 对 象 上 报告 的 所 有 警告 。 


@ void close() : 立即 释放 此 ResultSet 对 象 的 数据 库 和 JDBC 资源 ,而 不 是 等 待 对 


象 自动 关闭 时 才 释 放 它 们 。 


@ int findColumn(String columnName) : 将 给 定 ResultSet 列 名 映射 至 其 ResultSet 


列 索引 。 


第 七 步 ”关闭 创建 的 各 个 对 象 


操作 完成 以 后 ,要 把 所 有 使 用 的 JDBC 对 象 全 都 关闭 ,以 释放 JDBC 资源 。 关 闭 分 别 
使 用 各 自 的 close() 方 法 ,关闭 的 顺序 和 声明 顺序 相反 , 先 关 闭 结果 集 对 象 (如 果 有 ) ,再 关 


闭 语 句 对 象 ,最 后 关闭 连接 对 象 。 


下 面 给 出 课堂 练习 4-12 的 jsp 代码 部 分 的 示例 ,说 明 JSP 通过 JDBC-ODBC 桥 进行 


数据 库 操 作 的 步骤 。 
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闪 示 例 4-19 JSP 通过 JDBC-ODBC 桥 进行 数据 库 操作 


<%Q@page contentTYyPe= "text/html; charset=gb18030" language="]java"g> 
<%Q@page import="java.sql. * "和 > 


<HTML> 
Si 引入 java.sql 包 
<TITLE> 使 用 JDBC-ODBC 桥 访问 access 数据 库 < /TITLE> 
</HERAD> 
<BODY> 
< 


out.println ("<CENTER>"); 
out .println("<FONT size=4> 学 生 信 息 </FONT>"); 
Class .forName ("sun.jdbc.odbc.JdbcOdbcDriver"); 


载 入 驱动 程序 类 别 


String url="jdbc:odbc:Student"; 
String user=""; 设置 IDBC URL 


String pwd=""; 
Connection con=DriverManager.getConnection (url,user,pwd); 


获取 数据 库 连 接 


第 4 章 ”知识 准备 


PreparedStatement pstmt= con.prepareStatement ("SELECT * FROM student 
WHERE school =?", ResultSet. TYPE _ SCROLL _ INSENSITIVE, 


ResultSet .CONCUR READ ONLY); 
预 编译 SQL 语 句 


pstmt .setstring (1, "软件 学 院 ");， 传递 IN 参 数 


ResultSet rs=pstmt .executeQuery (); 


eR 执行 SQL 语 名， 获得 结果 集 
<TR> 
<TD align=center><B> 姓 名 </B></TD> 
<TD align=center><B> 学 院 </B></TD> 
<TD align=center><B> 年 龄 </B>< /TD> 
</TR> 
<% 
rs.beforeFirst () 二 二 将 游标 移 至 结果 集 第 一 行 之 前 
while (rs.next () ) 一 一 将 游标 从 当前 位 置 向 下 移动 一 行 | 
1 
%> 
<TR ALIGN=CENTER> -所 获取 当前 行 中 name 列 的 值 
<TD><%=rs.getstring ("name")®%></TD> 
<TD><%=rs.getString ("school1")%></TD> 一 一 获取 当前 行 中 school 列 的 值 
<TD><%=rs.getInt ("age")%></TD> 
TRS | 获取 当前 行 中 age 列 的 值 
rs.close(); 
pstmt .close () ;一 一 关闭 语句 对 象 
con.close(); 
> | 
</TABLE> 
</CENTER> 
</BODY> 
<HTML> 


向 数据 表 student 中 添加 三 条 记录 ,如 图 4-24 所 示 。 


2, 李 铭 | 财经 学 院 | 19 
3 王 圳 | 外 语 学 院 | 21 


图 4-24 student 数据 表 中 的 记录 


示例 上-19 中 的 search. jsp 执行 的 结果 是 把 软件 学 院 的 学 生 显示 在 浏览 器 中 ,如 图 25 
所 示 。 

为 了 便于 读者 理解 ,课堂 练习 4-12 演示 了 如 何 使 用 JDBC-ODBC 桥 的 方式 进行 数据 

库 的 一 般 性 操作 。 示 例 源 文件 中 所 采用 的 数据 库 的 连接 方式 ,是 对 每 一 次 数据 库 操作 请 

求 都 要 建立 一 个 数据 库 的 连接 ,对 资源 消耗 很 大 ,所 以 从 第 5 章 开始 ,数据 库 的 连接 技术 
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中 小 型 Web 项 目 开 发 实战 


SO |B hp/ocahostso80/searchjsp 7| [S| [x Es. an [加 


帘 安 大 使 用 DBC-ODBC 析 访问 ACCE.. 全 "上 国 7 了 往 了 愉 P)Y 何 工具 0)v” 


5 画 夯 | | 图 Internet | 全 PP 模式 : 用 [E100% = 


图 4-25 ”student 数据 表 中 的 记录 


将 采用 第 3 章 任 务 六 中 的 连接 池 方 式 。 


4.6.7 JDBC URL 


示例 4-19 中 连接 数据 库 时 使 用 了 JDBC URL。JDBC URL 是 一 种 标识 数据 库 的 方 


法 ,可 以 使 相应 的 驱动 程序 能 够 识别 该 数据 库 并 与 它 建立 连接 。 标 准 的 JDBC URL 的 格 
式 如 下 : 


jdbc : < 子 协议 名 > : < 子 名 称 > 
JDBC URL 由 三 个 部 分 组 成 ,各 个 部 分 之 间 用 冒号 分 隔 。 去 子 协 议 之 是 指数 据 库 连 


接 的 方式 。 去 子 名 称 二 可 以 根据 子 协议 的 改变 而 变化 。 
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JDBC-ODBC 桥 驱 动 程序 使 用 ODBC 子 协 议 。 该 子 协议 的 URL 格式 如 下 : 


jdbc:odbc:<data source name> [<attribute namel>=<attribute valuel>]... 
[<attribute namen>=<attribute valuen>] 


下 面 列 出 常用 数据 库 的 JDBC URL 的 格式 。 

(1) Microsoft SQL Server 2000 

驱动 程序 包 名 : msbase. jar mssqlserver. jar msutil. jar 

驱动 程序 类 名 : com. microsoft. jdbc. sqlserver. SQLServerDriver 
JDBC URL: jdbc:microsoft:sqlserver:// 一 server_name 二 :一 port 二 
默认 端口 为 1433。 如 果 服 务 器 使 用 默认 端口 , 则 port 可 以 省 略 。 

(2) Microsoft SQL Server 2005 JDBC Driver 

驱动 程序 包 名 : sqljdbc. jar 

驱动 程序 类 名 : com. microsoft. sqlserver. jdbc. SQLServerDriver 
JDBC URL: jdbe: sqlserver://=server_name> :<port> 

默认 端口 为 1433。 如 果 服 务 器 使 用 默认 端口 , 则 port 可 以 省 略 。 

(3) Oracle Thin JDBC Driver 

驱动 程序 包 名 : ojdbcl4. jar 

驱动 程序 类 名 : oracle. jdbc. driver. OracleDriver 

JDBC URL: jdbc:oracle:thin:@// 一 host> :二 port 二 /ServiceName 
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或 jdbc:oracle:thin:@<=host> :port> :二 SID 放 > 

(4) IBM DB2 Universal Driver Type 4 

驱动 程序 包 名 : db2jcc. jar db2jcc_license_cu. jar 

驱动 程序 类 名 : com. ibm. db2. jcc. DB2Driver 

JDBC URL: jdbc:db2:;//=<host>[:=<port>]/<database_name> 

(5) IBM DB2 Universal Driver Type 2 

驱动 程序 包 名 : db2jcc. jar db2jcc_license_cu. jar 

驱动 程序 类 名 : com. ibm. db2. jcc. DB2Driver 

JDBC URL: jdbc:db2 :一 database_name 二 

(6) MySQL Connector/J Driver 

驱动 程序 包 名 : mysql-connector-java-x. x. xx-bin. jar 

驱动 程序 类 名 : com. mysql. jdbc. Driver 

JDBC URL: jdbe:mysql://=host> :<port>/<database_name> 

默认 端口 为 3306。 如 果 服 务 器 使 用 默认 端口 , 则 port 可 以 省 略 。 

(7) Informix JDBC Driver 

驱动 程序 包 名 : ifxjdbc. jar 

驱动 程序 类 名 : com. informix.jdbc. IfxDriver 

JDBC URL: jdbc: informix-sqli: // (<ip-address> |=host-name>} : <port- 
number>[/<dbname>]: INFORMIXSERVER=<server-name> 

(8) Sybase Adaptive Server Enterprise JDBC Driver 

驱动 程序 包 名 : jconn2. jar 或 jconn3. jar 

驱动 程序 类 名 : com. sybase. jdbc2. jdbc. SybDriver 或 com. sybase. jdbc3. jdbc. SybDriver 

JDBC URL: jdbce: sybase: Tds:=host> :=port> 

默认 端口 为 5000。 

(9) Sybase Adaptive Server Anywhere or Sybase IQ JDBC Driver 

驱动 程序 包 名 : jconn2. jar 或 jconn3. jar 

驱动 程序 类 名 : com. sybase. jdbc2. jdbc. SybDriver 或 com. sybase. jdbc3. 
jdbc. SybDriver 

JDBC URL: jdbc: sybase: Tds: 一 host 二 :二 port 二 ? ServiceName= < database_ 
name> 

默认 端口 为 2638。 


课 后 练习 


1. 仿照 新 浪 会 员 中 心 注册 页 ,利用 HTML 表单 编写 一 个 注册 页 。 
2. 如 何 使 课堂 练习 3-2 输出 的 表格 中 的 相 邻 行 的 底 色 不 同 ? 试 实现 之 。 
提示 : 使 用 数组 元 素 表 示 颜 色 。 
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中 小 型 Web 项 目 开 发 实战 


. JSP 的 指令 标签 有 哪些 ?它们 的 语法 格式 是 怎样 的 ? 分 别 起 到 什么 作用 ? 

. JSP 的 动作 标签 有 哪些 ?它们 的 语法 格式 是 怎样 的 ?分 别 起 到 什么 作用 ? 
.session 对 象 的 工作 原理 是 什么 ? 

.session 对 象 和 application 对 象 有 什么 区 别 ? 

. 试 将 课堂 练习 4-12 中 的 Access 数据 库 蔡 换 为 MySQL 数据 库 , 查 询 年 龄 大 于 


20 岁 的 学 生 ,并 将 其 基本 信息 显示 在 浏览 器 中 。 
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系统 数据 访问 功能 模块 的 
设计 开发 


第 5 章 商品 信息 的 显示 和 查询 


本 章 学 习 要 点 : 

。 熟练 掌握 JavaBean 的 基本 要 素 ; 

熟练 掌握 实体 类 的 编写 规范 ; 

。 熟练 掌握 DAO 类 中 各 种 查询 方法 的 编写 技巧 ; 

。 熟练 掌握 SQL SELECT 语句 的 各 种 用 法 

熟练 掌握 一 jsp:useBean 盖 一 jsp:setProperty 二 一 jsp:getProperty 二 动作 标签 的 用 法 ; 
。 了 解 JavaBean 的 作用 范围 。 


5.1 任务 一 : 商品 展示 的 实现 


问题 : 使 用 JSP 十 JavaBean 技术 ,如 何 将 数据 表 中 的 信息 显示 到 浏览 器 中 呢 ? 使 用 
JavaBean 又 有 哪些 Java Web 程序 员 普 遍 接受 的 约定 需要 遵守 呢 ? 


”任务 目标 : 
| 将 第 3 章 中 创建 的 goods 表 中 的 记录 采用 JSP 十 JavaBean 的 技术 显示 在 浏览 
器 中 。 
”技能 训练 : 
: 。 实体 类 的 编写 规范 。 

。 DAO 类 findAll(O 〇 ) 方 法 的 编写 规范 。 

。 JSP 与 JavaBean 相关 的 动作 标签 的 使 用 。 


本 任务 中 提出 使 用 JSP 十 JavaBean 技术 来 完成 这 一 任务 .那么 什么 是 JavaBean 呢 ? 
5.1.1 JavaBean 的 定义 


JavaBean 一 种 满足 特殊 基本 要 素 的 Java 类 ,这 些 要 素 使 它们 可 以 很 容易 地 被 重复 使 
并 且 可 以 进行 组 合 ,从 而 可 以 应 用 在 不 同 的 应 用 程序 中 。 

JavaBean 的 基本 要 素 如 下 : 

(1) JavaBean 必须 放 在 一 个 包 (package) 中 ; 

(2) JavaBean 必须 有 一 个 不 带 参 数 的 构造 器 ; 


(3) Bean 类 应 该 没有 任何 公共 变量 ,Bean 类 的 成 员 变量 都 是 private 类 型 的 ; 

(4) 通过 public 的 getters 和 setters 方法 来 读 取 或 者 设置 成 员 变 量 的 值 ,getters 和 
setters 方法 和 对 应 的 实例 变量 之 间 的 命名 也 要 遵循 一 定 的 约定 : 成 员 变量 名 称 首 字母 必 
须 小 写 ,而 对 应 的 getter/setter 方法 采用 “骆驼 命名 法 "(CamelCase) 命 名 , 即 ; set/get 十 
将 变量 首 字母 改 成 大 写 。 例 如 ,有 一 个 实例 变量 We setter 方法 名 称 应 该 定义 成 
setAge() ,而 它 的 getter 方 法 名 称 应 该 定义 成 getAge()。 这 种 命名 规则 有 一 个 特例 ,如 
果 成 员 变量 是 boolean 类 型 的 ,那么 , 它 对 应 的 getter 方法 应 该 采用 “is”" 开 头 , 例 如 ,假设 
有 一 个 实例 变量 married 表示 “ 婚 否 ”(true 或 者 false) ,那么 , 它 对 应 的 getter 方法 名 称 应 
该 是 isMarried()。 而 不 是 getMarried(), 而 对 于 setter 方法 .规则 不 变 。 


5.1.2 商品 信息 实体 Bean 的 编写 


最 常用 的 是 用 实体 Bean 代表 关系 库 中 的 数据 ,一 个 简单 的 实体 Bean 可 以 定义 成 代 
表 数据 库 表 的 一 个 记录 ,其 名 字 一 般 与 所 对 应 的 数据 表 同 名 ,并 且 首 字母 大 写 ,其 成 员 变 
量 (也 称 为 属性 ) 的 名 字 和 类 型 与 数据 表 中 的 列 的 名 字 和 类 型 相同 。 实 体 类 一 般 被 创建 后 
放 在 entity 包 中 。 下 面 分 两 步 来 完成 第 一 个 JavaBean 的 编写 。 

第 一 步 ” 如 图 5-1 所 示 为 goods 表 创建 一 个 与 之 相对 应 的 实体 Bean ,并 根据 数据 表 
完成 属性 的 编写 。 


实体 Bean 应 该 实现 java.io.Serializable 接口 。Java 
的 “对 象 序列 化 "能 自动 补偿 操作 系统 方面 的 差异 ， 
实体 Bean 一 般 使 得 在 不 同 的 操作 系统 比如 Windwos 和 UNIX 之 间 
放 在 entity 包 中 传输 数据 时 不 会 因 系统 差异 而 产生 错误 


package com.jyw, www,enLIESL 


import java.io.Serializable; 
public class Goods implements Serializablet 


日 由 co je wm entity 
四 国 Goods. java 

i JRE Systef Library [Su 

-Bh Java EE 9 Libraries 


BS Tebhoot 
中 
goods 的 属性 与 第 3 章 中 各 个 属性 与 goods 数 据 表 中 的 
goods 数 据 表 的 名 字 相 列 的 名 字 和 类 型 相同 


同 ， 并 且 首 字母 大 写 
图 5-1 实体 bean 属性 的 编写 


第 二 步 ” 属 性 写 好 后 ,可 以 利用 MyEclispe 中 source>Generate Getters and Setters 
命令 来 自动 生成 getters 和 setters 方法 ,如 图 5-2 所 示 。 

在 图 5-3 所 示 的 对 话 框 中 色 选 需要 创建 setters 和 getters 的 属性 。 

完成 后 就 会 生成 如 下 所 示 的 一 个 实体 Bean。 

闪 示 例 S-1 Goods. java 


Package com.jyw.www.entity; 
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| 


Refactor Navigate Search Project MyEclipse Bun| 


(]) 单 击 Source 菜 单 


Togele Comment Ctrlt7 
Add Block Comment Ctrltshiftt, 
Renove Block Comment CtrltShiftt\ 
Generate Elenent Comment ttShifttJ 
Shift Right 

Shift Left 

Correct Indentation CtrltI 

了 ormat Ctrl+Shi ft 了 


Fornat Elenent 


Add Import Ctrl+Shift+ 
Organire Imports Ctrl+Shi ft+0 
Ef 人 iemb， 
(2) 单 击 菜单 项 中 的 i 
ean Up. 


Override/Implement 
Methods.… 命 令 


Generate Getters and Setters. 


图 5-2 自动 生成 setters 和 getters 方法 的 菜单 选项 


Generate Getters and Set 


(1) 单 击 Select All 按 钮 


Select 如 1 


Deselect All 
Select Setters 


DAlow setters for final fislds (renove ' fingl’ modifier fro 
Insertion point: 

| After color’ > 
Sert by: 

[rields in getter/setter pairs Ea 


Aceess modifier 

(2) 选中 public 单 选 按钮 二 -@ publie Oprotected = Odsfault Oprivate 
Osing 口 :mchronized 

口 cansrat， 


The format of the getters/setters may be configured on the 
Lode Templates preference page. 


ethod conments 


i 10 of 10 selected. 


(3) 单 击 OK 按 钮 


图 5-3 选择 需要 生成 getters 和 setters 方法 的 属性 


import java.io.Serializable; 


public class Goods implements Serializable{ 


private String id; // 货 品 编号 
private String name; // 货 品名 称 
private String image; // 货 品 图 片 
private Float price; // 销 售 指导 价 
private String color; // 货 品 颜色 
private String size; // 货 品 尺码 


public String getId() { 
return id; 

} 

public void setId (String id) { 
this.id=id; 
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中 小 型 Web 项 目 开 发 实战 


Public String getName () { 


return name; 


} 
public void setName (String name) {——— 
this.name=name; 


对 照 JavaBean 的 基本 要 素 ， 体 会 
getters 和 setters 方 法 的 编写 规则 


} 

public String getImage () { 
return image; 

} 

public void setImage (String image) { 
this.image=image; 

public Float getPrice() { 
return price; 

} 

Public void setPrice (Float price) { 
this.price=price; 

} 

public String getColor() { 
return color; 

} 

public void setColor (String color) { 
this.color=color; 

} 

public void setSize(String size) { 
this.size=size; 

} 

public String getSize() { 
return size; 

} 


5.1.3 DAO 类 中 findAll 方法 的 编写 


本 任务 的 目标 是 从 数据 表 中 取出 数据 并 显示 在 浏览 器 中 ,所 以 编写 与 数据 库 操作 相 
关 的 DAO :(Data Access Objects) 类 。DAO 的 成 员 方 法 主要 完成 对 数据 表 中 的 数据 做 
增 、 删 \ 改 、 查 等 操作 。DAO 类 一 般 被 创建 在 dao 包 中 ,其 命名 约定 是 “所 要 操作 的 数据 表 
的 名 字 十 DAO”, 如 图 5-4 所 示 。 

DAO 中 的 各 方法 的 命名 也 有 约定 ,实现 查询 操作 的 方法 一 般 以 find 开头 ,比如 读 取 
所 有 记录 的 方法 命名 为 findAll()。 以 下 示例 是 GoodsDAO 方法 中 findAll() 方 法 的 源 
代码 。 

闪 示 例 5-2 GoodsDAO 类 中 的 findAll() 方 法 


public ArrayList<Goods>findal1() { 
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对 goods 表 进行 操作 的 DAO 类 
DAO 类 一 般 放 在 dao 包 中 命名 为 GoodsDAO 


public class GoodsDAO { 


) 


-mM Java EE 5 Libraries 
由 色 WebRoot 


图 5-4 创建 GoodsDAO 类 


ArrayList<Goods>ret=new ArrayList<Goods> (); 从 连接 池 中 
ev 获取 数据 库 连接 


InitialContext ctx=new InitialContext (); Ze 


DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 


Connection conn=ds.getConnection(); 


一 执行 对 goods 表 的 查询 语句 
Statement stm=conn.createStatement (); 1 


ResultSet res=stm.executeQuery ("SELECT # FROM goods"); 
while (res.next()) { 

Goods goods=new Goods (); 

goods .setId(res.getString("id") ) 7 


goods .setColor (res .getString("color") ) 


goods .setName (res ‘getstring (“name™) js SR i 
goods.setImage (res.getString ("image")); 并 放 在 ArrayList 中 
goods .setPrice (res.getFloat ("price")); 
goods.setSize(res.getString("size")); 
ret .add (goods); 
} 
res.close (); 
stm.close () ;一 一 后 一 将 数据 库 连 接 放 回 池 中 | 
conn.close(); 
} catch (Exception e) { 
e.printSstackTrace (); 
} 将 人 数据 表 中 恋 出 的 记录 信息 返回 给 调用 者 


return ret; 


5.1.4 编写 显示 商品 信息 的 JSP 文件 


实体 类 和 DAO 类 写 好 后 ,还 要 编写 JSP 文件 来 显示 从 数据 表 中 读 取 的 记录 。JSP 
文件 创建 在 项 目的 WebRoot 目录 下 ,如 图 5-5 所 示 。 
在 接着 出 现 的 对 话 框 中 按照 图 5-6 的 指示 操作 。 
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中 小 型 Web 项 目 开 发 实战 


(2) 单 击 弹出 菜单 中 的 New 命 令 


Bw 
白 器 src 


晶 - comjyw.www.dao 


(1) 右 击 项 目的 申 三 Java 国 Repor web Project 
WebRoot 文 件 夹 一 


从 paste cuv @ Interface 
其 Dalete Delete 人 Sow folder 
BFolder 
Build Path | Gre 
Refactor Al+Shift+T 上 区 Applet 
Import.. 四 HTML (Advanced Templates) 


(3) 单 击 级 联 菜单 中 的 JSP 命 令 
图 5-5 在 项 目 中 创建 JSP 文件 


JSP Wizard 


File Path: /jyw/WebRoot Browse-| 


Fa 一 (1) 输入 JSP 文 件 的 名 字 ， 命 
ED 名 规则 是 能 “ 望 名 知 意 


Template to use: |Default JSP template S| 


(2) 单 击 Finish 按 钮 


图 5-6 为 JSP 文 件 命名 


示例 5-3 给 出 了 showGoods.jsp 的 源 代码 ,对 goods 表 中 所 有 的 商品 基本 信息 的 记 


录 每 4 个 为 一 行进 行 显示 。 
闪 示 例 5-3 showGoods. jsp 


<%@page contentType="text/html; charset=gb18030" language= "java"s> 


<%@page import="com.jyw.www.entity.* ,com.jyw.www.dao.* ,java.util.* oe 
<HTML> 


<BODY> Page 指 令 定 义 
<CENTER> 了 编码 集 ， 需 要 
商品 展示 引入 包 等 信息 
<TABLE border=1> 
<% 
GoodsDAO goodsDao=new GoodsDRO (); 调用 GoodsDAO 中 的 
List<Goods>goodsList=goodsDao.findAll (); 人 findAll() 方 法 ， 获 得 
Iterator it=goodsList.iterator(); goods 表 中 所 有 的 记录 信息 
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while (it.hasNext()) { 
$> 


<TR ALIGN=CENTER> 
<% 


pe 


使 用 双重 循环 实现 商品 
信息 每 4 条 在 一 行 的 显示 


for (int j=0; j<4; j++) { 
Goods goods= (Goods) it.next (); 
%> 
<TD> 
<TABLE border=1> 


<TR><TD 人 图 片 放 在 根 目录 的 images 子 目录 下 


<img src="/images/<%=goods.getImage()®%>" height=140 width=120> 


</TD></TR> 


<TR ALIGN=CENTER><TD colspan="3"><%=goods.getId()®%></TD></TR> 
<TR ALIGN=CENTER><TD colspan="3"><%=goods.getName ()%></TD></TR> 


<TR><TD><%=goods.getColor ()%></TD> 
<TD><%=goods.getSize()%></TD> 
<TD>¥ <%=goods.getPrice()%></TD> 

</TR> 
< /TABLE> 
</TD> 
<% 
if (!it.hasNext ()) 


break; 


%> 
</TABLE> 
</BODY> 

< /CENTER> 
</HTML> 


文件 编写 完成 后 ,就 可 以 按照 第 2 章 的 方法 启动 Web 服务 器 ,将 项 目 发 布 到 服务 器 


http://localhost:8080/showGoods. jsp 


商品 展示 
JYW12QKB2 JYW12CKY2 JYW12CEBO JYW12CEN1 
和 牛仔裤 和 牛仔裤 和 牛仔 入 千 仔裤 
芽 E- 沁 [98.0 用 芽 上 -区 [128.0 || 匿 |E- 沁 [对 158.0 水 尝 黑 [S-XL[¥228.0 


图 5-7 商品 信息 显示 效果 


上 ,再 在 浏览 器 URL 中 输入 正确 的 访问 地 址 ,就 可 以 看 到 如 图 5-7 的 显示 效果 了 。 
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5.2 任务 二 : 商品 库存 信息 的 排序 显示 


问题 : 在 现实 生活 中 ,为 了 方便 检索 ,常常 会 对 数据 的 集合 进行 排序 ,比如 * 花 名 册 ” 

是 按照 同学 们 的 学 号 进行 了 排序 。 那 么 如 果 和 希望 对 任务 一 中 的 结果 进行 排序 显示 ,使 用 
JavaBean 技术 如 何 实现 呢 ? 

“任务 目标 : 
| 将 第 3 章 中 创建 的 stock 表 中 的 记录 采用 JSP 十 JavaBean 的 技术 显示 在 浏览 器 

; 中 ,并 按照 分 店 号 进行 排序 。 

技能 训练 ， 

。 完善 实体 类 的 编写 规范 。 

完善 DAO 类 findAll() 方 法 的 编写 规范 。 
。 JSP 与 JavaBean 相关 的 动作 标签 的 用 法 。 


任务 二 要 求 按照 分 店 号 及 stock 表 中 的 storenum 列 进行 排序 ,在 完成 任务 一 后 ,得 
到 了 所 有 商品 基本 信息 的 列表 ,在 没有 指明 按照 某 个 列 排序 的 情况 下 ,MySQL Select 默 
认 排 序 方法 是 按照 物理 存储 顺序 显示 。 


5.2.1 对 视图 的 排序 查询 


首先 要 在 项 目的 entity 包 中 添加 一 个 与 stock 表 相 对 应 的 实体 Bean Stock, 请 读者 


自行 完成 。 
然后 在 DAO 包 中 添加 StockDAO 类 ,其 中 的 findAll() 方 法 可 以 完成 排序 查询 ,需要 
用 SELECT 语句 的 ORDER BY 子 句 来 指明 按照 哪个 列 进行 排序 ,其 语法 规则 是 ， 


SELECT FROM 数据 表 的 名 字 ORDER BY 列 的 名 字 
闪 示 例 5-4 ”StockDAO 类 中 的 findAl10 〇 方法 


public List<Stock>findAll() { 
String selectsql="SELECT * FROM stock ORDER BY storenum "; 
List<Stock>ret=new ArrayList<Stock> (); 


按照 分 店 号 即 storenum 列 排序 


try{ 
InitialContext ctx=new InitialContext (); 


DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 


Connection conn=ds.getConnection(); 
Statement stm=conn.createStatement (); 
ResultSet res=stm.executeQuery (selectsql); 
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while (res.next()) { 
Stock stock=new Stock () ; 
stock.setId(res.getInt ("id")); while 循 环 将 stock 表 中 
stock.setGoodsid (res .getString("goodsid")) 7 的 记录 从 结果 集中 逐条 


stock.setSize (res.getString ("size")); a ie 
中 ， 再 返回 给 调用 方 


Stock .SetStock (es .getInt ("stock")); 


stock .setStorenum (res .getInt ("storenum")); 
ret.add (stock); 
} 
res.close(); 
stm.close (); 
conn.close(); 
} catch (Exception e) { 
e.printStackTrace (); 
} 


return ret; 


5.2.2 在 JSP 中 使 用 增强 的 FOR 循环 


Jdk1.5 增加 了 增强 的 for 循环 ,使 得 遍历 数组 或 集合 时 更 加 简便 。 其 语法 规则 如 下 : 
遍历 集合 的 语法 为 : 


for (Type value : Iterable) { 
expression value; 


} 
遍历 数组 语法 为 : 


for (Type value : array) { 
expression value; 
} 


示例 5-5 的 showStock. jsp 中 使 用 增强 的 for 循环 显示 Java ArrayList 中 的 信息 。 
闪 示 例 5-5 showStock. jsp 


<%@page contentType="text/html; charset=gb18030" language="java"®> 
<%@page import="com.jyw.www.entity.* ,com.jyw.www.dao.* ,java.util.* "$> 
<HTML> 
<HEAD><TITLE> 利 用 JDBC-ODBC 桥 访问 ACCESS 数据 库 < /TITLE>< /HEAD> 
<BODY> 
<TABLE border=1><TR><TD align=center><B> 序 号 </B></TD> 
<TD align=center><B> 商 品 编号 </B>< /TD> 
<TD align=center><B> 尺 码 </B></TD> 
<TD align=center><B> 库 存量 </B></TD> 
<TD align=center><B> 分 店 编号 </B>< /TD> 
</TR> 
< 


StockDAO stockdao=new StockDAO(); 
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中 小 型 Web 项 目 开 发 实战 


List<Stock> stocklist=stockdao.findAll (); 
for (Stock stock: stocklist) { 

$$> 

< TR RLIGN= CENTER> Bx 增强 的 for 循 环 ， 对 集合 进行 遍历 

<TD><%=stock.getId()®%></TD> 
<TD><%=stock.getGoodsid()%></TD> 
<TD><%=stock.getSize()®%></TD> 
<TD><%=stock.getStock()%></TD> 
<TD><%=stock.getStorenum()®%></TD> 


</TR> 
<% } %> 
< /TABLE> 从 s/ocshost :e000/shonstock jsp | 
< /BODY> 库存 商品 列表 
> 厚 号 [商品 编 寻 
三 ~ pe IJYW12CKBO 
如 图 5-8 所 示 为 showStock. jsp 执行 的 结果 。 a 
从 运行 结果 中 可 以 看 到 ,列表 按照 分 店 编号 从 [TIN12CEY2 
低 到 高 进行 了 排序 。 如 果 和 希望 列表 按照 分 店 号 从 高 Te 


到 低 进 行 排序 ,也 就 是 “ 反 序 ” ,就 需要 在 ORDER BY 时 
子 句 后 面 加 关键 字 DESC, 如 下 所 示 。 图 5-8 ”库存 商品 信息 的 显示 效果 


String selectsql=" SELECT * FROM stock ORDER BY storenum DESC" 


DESC 表 明 反 序 
显示 查询 结果 


5.3 任务 三 : 查询 各 分 店 的 库存 商品 详细 信息 


问题 : 我 们 在 购物 的 时 候 常 常会 遇 到 这 样 的 情况 : 想 要 购买 某 款 服装 或 者 鞋子 ,但 
摆 放 在 货架 上 的 商品 并 不 适合 我 们 想 要 的 尺码 ,这 时 ,店员 会 去 查询 店 里 是 否 还 有 适合 我 
们 想 要 的 尺码 的 库存 。 这 个 功能 采用 JSP 十 JavaBean 技术 是 如 何 实现 的 呢 ? 


| 查询 用 户 指定 店 的 库存 商品 ,包括 图 片 、 指 导 价 格 等 的 详细 信息 。 
”技能 训练 : 
: 。 视图 对 应 的 实体 类 的 编写 规范 。 

。 DAO 类 findByXxxx() 方 法 的 编写 规范 。 

完成 了 第 3 章 的 任务 四 后 ,获得 了 由 基 表 goods 和 stock 生成 的 视图 stockinfo, 这 个 
视图 中 汇总 了 任务 三 要 求 查询 的 所 有 信息 ,视图 可 以 看 做 是 “虚拟 的 表 ”,Java 对 数据 库 
中 视图 的 访问 ,可 以 使 用 访问 数据 表 的 技术 来 完成 。 

与 编写 对 数据 表 访 问 的 代码 相似 ,首先 编写 与 stockinfo 视图 对 应 的 实体 类 ,请 读者 
参照 前 面 的 小 节 自 行 完成 ,然后 编写 StockInfoDAO 类 中 的 方法 。 这 里 需要 根据 用 户 指 
定 的 店 来 查询 该 店 的 库存 ,这 是 一 个 条 件 查询 。 
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5.3.1 编写 DAO 类 中 的 findByXxxx 方法 


条 件 查询 方法 的 命名 约定 为 findByXxxx(Xxxx xxxx) ,例如 按照 分 店名 进行 查找 的 
方法 命名 为 findByStoreName (String storename)。 示 例 5-6 为 StockInfoDAO 类 中 的 
findByStoreName() 方 法 的 代码 。 

闪 示 例 5-6 ”StockInfoDAO 中 用 于 按 分 店 编号 查询 的 findByStoreName() 方 法 


public List<StockInfo>findByStoreName (String storename) { 将 用 户 指定 的 分 店 
List<StockInfo>ret=new ArrayList<StockInfo> () 7 名 作为 入 口 参 数 
tryt 

InitialContext ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 


Connection conn=ds.getConnection(); 

PreparedStatement pstmt=conn.prepareStatement ("SELECT * FROM stockinfo 
WHERE storename=?"); 

storename=new String (storename .getBytes ("iso- 8859-1"),"GB18030"); 


Pstmt .setString(1，storename) 预 编译 带 条 件 的 SELECT 语句 


ResultSet res=pstmt .executeQuery () 
while (res.next()) { 
StockInfo goods=new StockInfo () 
goods .setId(res.getString("id") ) 7 
goods .setName (res.getString ("name")); 
goods .setSize(res.getString("size"))7 
goods .setColor (res.getString ("color")); 
goods .setImage (res .getString("image") ) 7 
goods .setPrice (res.getFloat ("price")); 
goods .setStock (res .getInt ("stock")); 
goods .setStorename (storename); 
ret .add (goods); 
} 
res.close(); 
pstmt.close(); 
conn.close(); 
} catch (Exception e) { 
e.printstackTrace (); 
} 


return ret; 


5.3.2 编写 与 用 户 查 询 相 关 的 JSP 文件 


下 面 编写 showStockInfo. jsp 文件 完成 用 户 选 择 分 店名 和 根据 用 户 的 选择 显示 查询 
结果 的 功能 。 
闪 示 例 5-7 showStockInfo. jsp 中 用 于 查询 的 表单 部 分 如 下 


<%@page contentType="text/html; charset=gb18030" language="java"®> 
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| Web 项 目 开发 实战 


<%@page import="com.jyw.www.entity.* ,com.jyw.www.dao.* ,java.util.*"$%> 
<HTML> 
<BODY><CENTER> 
库存 商品 列表 
<form action="searchStock.jsp" method= "post"> 
店名 <select name="storenum"> 
<option></option> 
<option value="1"> 佳 衣 屋 海岸 城 店 < /option> 
</select> 
<input name="submit" type="submit" value=" 查 询 库存 "> 
</form> 


表单 部 分 在 IE 浏览 器 中 的 显示 结果 如 图 5-9 所 示 。 


二 下 | 科 hp /aocahostenao/awstocamto ji 国 | 加 于 玫 | 
库存 商品 列表 


图 5-9 用 户 选择 分 店 的 列表 菜单 


闪 示 例 5-8 ”showStockInfo. jsp 中 显示 查询 结果 的 部 分 如 下 : 


<TABLE border=1> 
< 
List<StockInfo>goodsList=stockinfoDAO. findByStoreName 
(request .getParameter ("storename")); 
Iterator it=goodsList.iterator () 7 


while (it.hasNext()) { 
2 以 用 户 选 择 的 分 店名 作为 入 口 参数 ， 
调用 示例 5-6 中 的 findByStoreName() 方 法 


<TR ALIGN=CENTER> 


<% 
for {int j=07 j<4; j++) { 
StockInfo goods= (StockInfo) it.next (); 
和 > 
<TD> 
<TABLE border=1> 
<TR><TD colspan="4"> 
<img src="/images/<%=goods.getImage()$>" height=220 width=180> 
</TD>< /TR> 
<TR ALIGN=CENTER><TD colspan="4"><%=goods.getId()%></TD></TR> 
<TR ALIGN=CENTER><TD colspan="4"><$%=goods .getName ()%></TD></TR> 
<TR><TD><%=goods.getSize()®%></TD> 
<TD><%=goods.getColor()%></TD> 
<TD>Y¥ <%-goods.getPrice () s></TD> 
<TD><%=goods.getstock()%></TD> 
</TR> 
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</TABLE> 
</TD> 
< 
if (!it.hasNext ()) 
break; 
} 
多 > 
</TR> 
< 
} 
和 > 
</TABLE> 


查询 “ 佳 衣 屋 总 店 华强 北 店 ”的 库存 商品 详细 信息 的 结果 如 图 5-10 所 示 。 


[gS htplocalhost8oso/showstocktnfojsp?storename=%BC%D1%6D2s6C236CE%6DD3%6D7%6DC 习 回国 [ae . mns 


久 http://localhost:s080/showstockinfojsp?stor. 僵 ~ 加 -机 ~ 芭 7Zmp vv 他 


库存 商品 列表 
选择 分 让 ” 导 国 囊 


下 


JYWI2CKBO | JYW12CKBO JYW12CKN1 JYW12CKY2 

| 牛仔 裤 | 和 牛仔裤 外 牛仔 裤 | 牛仔 裤 

上 | 佳 衣 屋 总 店 华强 北 店 ||| 佳 衣 屋 总 店 华强 北 店 || 佳 衣 屋 总 店 华强 北 店 || 佳 衣 屋 总 店 华强 北 店 
Iw 芽 人 158.0 |40 | [五 | 广 158.0 |20 | 的 洗 村 这 228.050 几 | 贰 公 128.0 [100 


图 5-10 查询 指定 分 店 的 库存 商品 信息 


5.4 任务 四 : 用 多 个 条 件 查 询 库 存 商 品 信 息 


问题 : 我 们 在 购物 的 时 候 还 会 遇 到 这 样 的 情况 : 我 们 想 要 购买 的 某 款 服装 或 者 鞋子 
的 尺码 在 当前 这 个 分 店 已 经 售 完 ,这 时 店员 常常 会 去 查询 其 他 分 店 是 否 还 有 我 们 想 要 的 
款式 和 尺码 的 商品 ,这 个 查询 功能 使 用 JSP 十 JavaBean 技术 是 怎样 实现 的 呢 ? 


We : 
”用 户 可 以 指定 商品 编号 .名称 、 颜 色 、 尺 码 、 库 存量 中 的 一 个 或 多 个 条 件 查询 库存 
商品 (包括 图片 时 从 入 人] 的 有 二 信 起 关中 商品 者 称 是 抱 棋 查 光 5 
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中 小 型 Web 项 目 开 发 实战 


”技能 训练 : 
: 。 视图 对 应 的 实体 类 的 编写 规范 。 
。 DAO 类 findByExample() 方 法 的 编写 规范 。 


5.4.1 编写 DAO 类 中 的 findByExample 方法 


任务 四 要 求 完成 一 个 多 条 件 的 查询 ,在 编写 条 件 SELECT 语句 时 首先 应 明确 用 户 指 
定 的 条 件 是 哪 几 个 ,将 用 户 指 定 的 条 件 写 在 SELECT 语句 的 WHERE 子 句 中 , 即 可 以 构 
成 一 个 完整 的 可 执行 的 SELECT 语句 ,并 将 查询 结果 返回 。 可 以 在 StockInfoDAO 中 添 
加 一 个 findByExample() 方 法 实现 此 功能 ,此 方法 需要 一 个 实体 Bean 类 型 的 入 口 参数 。 
闪 示 例 5-9 StockInfoDAO 中 的 findByExample() 方 法 


public List<StockInfo>findBYExample (StockInfo goods) { 


String selectsql=" SELECT * FROM AS 入 口 参数 为 StockInfo 
Boolean isFirstCondition=true; 类 型 的 Bean，Bean 中 值 为 null 


if (goods.getId() !=null) { 的 属性 表示 不 是 查询 条 件 
selectsql=selectsql+" WHERE "+" id='"+goods.getId()+"'"; 


isFirstCondition=false; 
} 逐个 判断 入 口 参数 中 各 个 属性 
是 否 为 null， 不 为 null 的 属性 要 
String name=goods .getName (); 添加 到 WHERE 子 句 中 作为 一 个 条 件 
if (name !=null) { 


try { 
name=new String (name .getBytes ("ISO- 8859-1")); 
} catch (UnsupportedEncodingException el) { 
// TODO Auto-generated catch block 
el.printStackTrace (); 
} 如 果 是 第 一 个 条 件 ， 前 面 须 加 WHERE 关键 字 
if (isFirstCondition) { 
selectsql=selectsql+" WHERE "+" name LIKE '%"+name+ "$$'"; 


1 397StConaitionm false; [如果 不是 第 一 个 条 件 ， 与 前 而 条 件 问 须 加 and 关 键 字 


} else 
selectsql=selectsql+" RND "+" name LIKE '%"+namet+ "%®'"; 
3 
if (goods.getSize() !=null) { 要 求 针对 货品 名 进行 模糊 查询 


if (isFirstCondition) { 


selectsql=selectsql+" WHERE "+" size='"+goods.getSize()+"'"; 
isFirstCondition=false; 

} else 

selectsql=selectsql+" AND"+" size='"+goo0ds.getSize()+""'"; 


J 
String color=goods .getColor (); 
if (color !=nu11) { 有 中 文 参数 的 ， 上 进行 编 色 转换 


ew 
color=new String (color.getBytes ("ISO- 8859-1")); 


100 


息 的 显示 和 查询 


} catch (UnsupportedEncodingException e) { 
// TODO Auto- generated catch block 
e.printStackTrace (); 
} 
if (isFirstCondition) { 
selectsql=selectsql+" WHERE "+" color="'"+colort""'™"; 
isFirstCondition=false; 
} else 
selectsql=selectsql+"AND "+" color="'"+color+ 


} 
Integer stock=goods.getstock(); 
if (stock !=nu11) { 
switch (stock) 1 == 对 于 列表 菜单 提交 的 参数 ， 用 switch case 语 句 来 处 理 
Case 1: 
if (isFirstCondition) { 
selectsql=selectsql+" WHERE "+" stock<=20"; 


isFirstCondition=false; 
} else 
selectsql=selectsql+" AND "+" stock<=20"; 
break; 
case 2: 
if (isFirstCondition) { 
selectsql=selectsql+" WHERE "+" stock>20 and stock<=50"; 
isFirstCondition=false; 
} else 
selectsql=selectsql+" AND "+" stock>20 and stock<=50"; 
break; 
Case 3: 
if (isFirstCondition) { 
selectsql=selectsql+" WHERE "+" stock>50"; 
isFirstCondition=false; 
helse 
selectsql=selectsql+" AND "+" stock>50"; 
break; 


List<StockInfo>ret=new ArrayList<StockInfo> (); 
try { 
InitialContext ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds .getConnection (); 
Statement stm= conn.createStatement (); 
ResultSet res=stm.executeQuery (selectsql); 
while (res.next()) { 
StockInfo fGoods=new Goods (); 
fGoods .setId (res .getString("id"))7 
fGoods .setName (res .getString("name") ) ; 
fGoods .setSize (res.-getString("size")) 7 
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中 小 型 Web 项 目 开 发 实战 


fGoods .setColor (res .getString("color")) 7 
fEGoods .setImage (res .getString("image")) 7 
fGoods .setPrice (res .getFloat ("price")); 
fGoods .setStock (res .getInt ("stock")); 
ret.add(fGoods); 


res.close(); 
stm.close(); 
conn.close (); 

} catch (Exception e) { 
e.printstackTrace(); 


return ret; 


5.4.2 编写 与 用 户 多 条 件 查 询 相 关 的 JSP 文件 


下 面 编写 用 户 输入 或 者 选择 查询 条 件 的 JSP 页 面 。 
关 示 例 5-10 searchStock. jsp 中 用 于 查询 的 表单 部 分 如 下 : 


<%Q@page contentType="text/html; charset=gb18030" language="java"s®> 
查询 结果 
<form action="searchStock.jsp" method="post"> 
商品 编号 <input name="id" type="text"> 
商品 名 称 <input name="name" type="text"> 
颜色 <input name="color" type="text" size="10"> 
尺码 <select name="size"> 
<option></option> 
<option>S</option> 
<option>M< /option> 
<option>L< /option> 
<option>XL< /option> 
</select> 
库存 量 < select name= "stock"> 
<option></optiony> 
<option value="1">20 以 下 </option> 
<option value="2">20- 50< /option> 
<option value="3">50 以 上 </option> 
</select> 
<input name="submit" type="submit" value= "查询 "> 
</form> 


在 I 浏览 器 中 的 显示 结果 如 图 5-11 所 示 。 

用 户 提 交 的 条 件 就 是 查询 条 件 ,根据 第 4 章 中 对 JSP 内 置 对 象 request 的 介绍 可 
以 知道 ,通过 request. getParameter() 方 法 可 以 读 取 用 户 通 过 表单 元 素 提 交 的 参数 ， 
再 调用 实体 Bean 的 setters 方 法 并 用 用 户 提 交 的 参数 来 设置 Bean 的 相应 属性 。 但 是 ， 
<jsp:useBean 这 和 <<jsp:setProperty 这 动作 标签 提供 了 更 加 便捷 的 方法 来 完成 这 一 操作 。 
102 


注意 : 为 了 方便 处 理 ， 应 将 库 
存量 选项 的 值 设置 为 整数 


查询 结果 
商品 编号 三 商品 名 称 [ 颜色 [ 尺码 区 回 库存 是 一 梧 匿 囊 


图 5-11 不 确定 多 条 件 的 用 户 查 询 界面 


5.4.3 用 jsp:useBean 创建 一 个 Bean 实例 


在 JSP 页 面 中 可 以 使 用 一 jsp:useBean 过 动作 来 创建 一 个 Bean 实例 。 志 jsp:useBean 过 
的 语法 规则 如 下 。useBean 的 定义 方式 有 两 种 ,分 别 如 下 面 的 两 段 代码 所 示 。 


<jsp:useBean 

id="beanInstanceName" 

scope="page | request | session | application" 

{ 
class="package.class" | 
type="package.class" | 
class="package.class" type="package.class" | 
beanName="[package.class | <%=expression %>]" type="package.class" 


/> 
和 


<jsp:useBean 
id="beanInstanceName" 
scope="page | request | session | application" 
{ 
class="package.class" | 
type="package.class" | 
class="package.class" type="package.class" | 
beanName=" [package.class | <%$=expression %>]" type="package.class" 
}> 
<jsp:setProperty.../> 


<jsp:useBean/> 
二 jsp:useBean 之 起 始 标记 和 结束 标记 之 间 的 语句 只 有 在 新 建 JavaBean 实例 的 时 候 
才 会 执行 ,如 果 是 使 用 已 经 存在 的 JavaBean 实例 , 则 不 会 执行 。 第 二 种 方式 可 用 于 初始 
化 一 些 属 性 。 
注意 : 一 jsp :setProperty 二 可 以 单独 放 到 外 面 ,并 独立 于 一 jsp :useBean 二 而 存在 。 
下 面 是 对 useBean 动作 属性 的 简要 说 明 。 
。 scope: 用 于 指定 JavaBean 的 作用 范围 ,可 能 的 取 值 为 page、 request、 session、 
application, 它 们 的 作用 范围 从 小 到 大 。 
。 id: 定义 Bean 实例 的 名 字 。 在 scope 定义 的 范围 内 ,使 用 此 名 字 来 分 辨 不 同 的 
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Bean。 这 个 变量 名 对 大 小 写 敏感 。 
。 class: 使 用 new 关键 字 以 及 class 构造 器 来 创建 指定 类 的 bean 实例 。 这 个 类 不 
能 是 抽象 的 ,必须 有 一 个 公用 的 、 没 有 参数 的 构造 器 , package 的 名 字 对 大 小 写 
。 type: 指定 JavaBean 对 象 的 类 型 ,通常 用 在 当 对 应 的 scope 中 已 经 存在 JavaBean 
对 象 的 时 候 , 这 个 时 候 使 用 type 将 不 会 产生 新 的 对 象 。 另 外 , 它 也 可 以 和 class 
属性 结合 使 用 ,此 时 ,type 指定 的 类 可 以 和 class 指定 的 类 是 同一 个 类 ,或 者 是 
class 指定 类 的 父 类 、 父 接口 。 
。 beanName: 使 用 java. beans. Beans. instantiate( ) 方 法 ,从 一 个 类 或 者 一 个 系列 模 
板 来 实例 化 一 个 JavaBean ,并 且 指 定 JavaBean 的 类 型 为 type 属性 指定 的 类 型 。 
这 个 属性 通常 并 不 使 用 。 
注意 : 使 用 jsp:useBean 可 以 得 到 一 个 JavaBean 的 实例 ,但 是 ,这 个 实例 不 一 定 是 新 
建 的 ,如 果 已 经 存在 一 个 与 指定 id 同名、 相同 scope 的 JavaBean, 那么 就 不 会 新 建 一 个 
JavaBean 实例 ,而 是 直接 使 用 已 经 存在 的 实例 。 
在 searchStock. jsp 中 ,添加 下 面 语句 来 获得 一 个 StockInfo 类 型 的 实体 Bean 实例 
stockinfo 。 


< jsp:useBean id="stockinfo" scope="request" class="com. jyw. www. entity. 
StockInfo"/> 


5.4.4 jsp :setProperty 关联 查询 参数 与 实体 Bean 的 
属性 


如 果 请 求 参 数 和 JavaBean 的 属性 名 称 一 致 ,可 以 使 用 下 面 的 方式 将 所 有 的 属性 和 相 
同名 称 的 请 求 参数 关联 起 来 : 只 需要 在 setProperty 中 将 * ”作为 property 的 值 : 


<jsp:setProperty name=" stockinfo " property="*"/> 


通过 这 种 方式 ,可 以 非常 简便 地 将 JavaBean 的 属性 和 请 求 参数 关联 起 来 。 如 果 有 部 
分 的 JavaBean 属性 无 法 找到 对 应 的 请 求 参数 , 则 不 会 采取 任何 行动 。 
关 示 例 5-11 ”searchStock. jsp 中 关联 表单 参数 与 实体 Bean 属性 的 部 分 代码 如 下 : 


下 面 的 代码 中 要 用 到 List， 所 以 引入 了 java.util 包 

<%@page import="java.util.* ,com.jyw.www.entity.* "%> 
2 使 用 useBean 动 作 产生 的 StockInfo 类 型 的 实例 stockinfo 
<jsp:useBean id="stockinfo" scope="request" class="com.jyw.www.entity.StockInfo"/> 


使 用 setProperty 动 作 关联 request 域 中 
和 的 表单 元 素 参数 和 goodsBean 的 属性 


<jsp:setProperty name="stockinfo" property="* "/> 
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使 用 useBean 动 作 产生 StockInfoDAO 
类 型 的 实例 stockinfoDAO 


<jsp:useBean id="stoockinfoDAO" scope="session” 


class="com.jyw.www.dao.StockInfoDAO"/> 


再 在 searchStock. jsp 中 添加 查询 结果 的 显示 部 分 。 
闪 示 例 5-12 ”searchStock. jsp 中 显示 查询 结果 的 部 分 代码 如 下 。 


<TABLE border=1> 


二 和 
List<StockInfo>goodsList=stockinfoDAO.findByExample (stockinfo); 
Iterator it=goodsList.iterator (); 一 > 
while (it.hasNext ()) { 人 ei A 
<TR ALIGN=CENTER> 
<% 
for (int j=0; j<4; j++) { 
StockInfo goods= (StockInfo) it.next (); 
多 > 
<TD> 
<TABLE border=1> 
<TR><TD colspan="4"> 
<img src="/images/<%=goods .getImage ()%>" height=220 width=180> 
</TD>< /TR> 
<TR ALIGN=CENTER><TD colspan="4"><%=goods.getId()®%></TD></TR> 
<TR ALIGN=CENTER><TD colspan="4"><%=goods.getName ()%></TD></TR> 
<TR><TD><%=goods.getSize()%></TD> 
<TD><%=goods.getColor ()®%></TD> 
<TD> ¥ <%=goods.getPrice()%></TD> 
<TD><%=goods.getStock()®%></TD> 
</TR> 
< /TABLE> 
</TD> 
<% 
if (!it.hasNext ()) 
break; 
} 
%> 
</TR> 
< 
} 
多 > 
</TABLE> 


在 无 任何 查询 条 件 的 情况 下 ,searchStock. jsp 的 运行 结果 为 显示 所 有 库存 ,如 图 5-12 
所 示 。 
以 商品 编号 为 JYW12CKN1 和 尺码 为 M 作为 条 件 的 查询 结果 如 图 5-13 所 示 。 
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计 右 大 htpy/localhost8080/searchstockjsp [ 画 [ | 
库存 查询 结果 
商品 编号 [ 商品 名 称 [ 颜色 [ 尺码 三 司 库存 量 [四 要 下 


JYW12CKBO JYWI2CKBO || JYW12CKNI JYW12CKN1 
牛仔 裤 牛仔 裤 牛仔 裤 牛仔 裤 
佳 衣 必 总 让 华强 北 店 || 佳 衣 屋 总 店 华强 北 店 || 佳 衣 屋 总 店 华强 北 店 | | 佳 衣 尾 海岸 城 让 
MI 匡 [158.0 [40 | 几 s 芽 [ 肥 158.0 [20 | 品 永 洗 昌 [这 228. 050| 冉 水洗 时 [ 宇 228.0 有 8 


JYW12CKY2 
年 三 入 
住 阴 尾 各 店 笠 强 北 让 


图 5-12 首次 进入 searchStock. jsp 页 的 效果 示意 图 


左下 二 httpy/localhostao80/searchstockjsp 三 了 旧 7 屈 v 加 EP 本 了 NO) 
库存 查询 结果 


商品 编号 商品 名 称 [ 颜色 | 尺码 六 辐 库存 量 [ ” 回 本 蔬 


5.5 知识 扩展 


5.5.1 JavaBean 的 范围 


在 5.4. 3 小 节 中 ,使 用 一 jsp:useBean 盖 动作 标签 元 素来 声明 一 个 在 某 一 个 范围 内 可 
存 取 的 JavaBean, 这 个 范围 可 以 是 application ,session request 和 page 中 的 一 个 ,并 通过 
scope 来 设置 。 
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page 是 默认 的 JavaBean 范围 ,这 种 范围 的 JavaBean 只 能 在 实例 化 它 的 页 面 中 使 用 ， 
此 时 ,JSP 动作 定义 的 JavaBean 实例 会 被 保存 到 PageContext 对 象 中 。 

当 属 性 scope 的 值 为 request 时 ,JSP 动作 定义 的 JavaBean 实例 就 会 存储 到 
ServletRequest 对 象 中 。 在 这 种 情况 下 ,此 实例 可 以 被 包含 它 的 JSP 页 面 所 调用 ,不 同 的 
语句 可 以 共享 同一 个 request 对 象 ,请 求 结束 ,该 实例 的 生命 周期 即 结束 。 使 用 jsp: 
include jsp:forward 或 者 RequestDispatcher 的 include() 方 法 或 者 forward() 方 法 时 ,两 
个 JSP 页 面 或 者 Servlet 页 面 之 间 将 会 共享 这 个 JavaBean 对 象 。 

当 属 性 scope 的 值 为 session 时 ,JSP 动作 定义 的 JavaBean 实例 就 会 存储 到 
HttpSession 对 象 中 ,可 通过 HTTP 请 求 使 用 。 该 JavaBean 实例 会 在 整个 会 话 (session) 
过 程 中 存在 。 

当 属性 scope 的 值 为 application 时 ,JSP 动作 定义 的 JavaBean 实例 就 会 被 存储 到 
ServletContext 对 象 中 。 这 意味 着 JavaBean 实例 可 以 被 运行 在 应 用 服务 器 的 在 相同 生命 
周期 中 的 任何 对 象 (例如 JSP 和 Servlet) 调 用 。 


5.5.2 使 用 jsp:setProperty 关联 Bean 属性 和 request 
参数 


在 5. 4.4 小 节 中 使 用 二 jsp :setProperty 过 动作 在 表单 的 参数 和 实体 Bean 的 属性 间 
做 了 批量 关联 。 志 jsp:setProperty 之 动作 还 有 其 他 多 种 用 法 ,读者 可 以 根据 业务 逻辑 实 
现 的 需要 进行 适当 地 选择 。 表 5-1 概述 了 使 用 二 jsp:setProperty 过 元 素来 设置 JavaBean 
组 件 的 属性 的 各 种 方法 。 


表 5-1 用 String 值 给 Bean 的 属性 值 赋值 
值 的 来 源 元 素 语 法 
a 一 jsp:setProperty name= "beanName" 
String 常量 a " i " 
property= "propName" value= "string constant" /> 
=jsp:setProperty name= "beanName" 
Request 参数 ee 
property 一 "propName”param 一 "paramName" /二 
一 jsp:setProperty name 一 "beanName” property 一 "propName" /二 
或 者 


一 jsp:setProperty name 一 "beanName” property="*"/> 


匹配 bean 属性 的 
Request 参数 


一 jsp:setProperty name= "beanName" 
property= "propName" value= "expression"/> 
或 者 


=ijsp:setProperty name= "beanName" property= "propName"> 


表达 式 =jsp:attribute name= "value"> 
expression 
=/jsp:attribute> 
一 /jsp:setProperty 二 
在 使 用 一 jsp:setProperty 二 元 素来 设置 JavaBean 组 件 的 属性 时 ,要 注意 以 下 几 点 。 


(1) 在 使 用 这 个 元 素 之 前 必须 使 用 二 jsp:useBean 记 声明 此 Bean。 
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(2) 一 jsp:setProperty 二 中 的 beanName 必须 和 useBean 属性 里 面 定义 的 id 属性 值 
相同 。 

(3) 在 JavaBean 组 件 中 必须 有 一 个 相应 的 setPropertyName 方法 。 

从 一 个 字符 串 常 量 或 者 请 求 参数 来 设置 属性 时 需要 用 到 下 表 中 列 出 的 类 型 。 如 果 常 
量 或 者 请 求 (request) 参 数 是 字符 串 (String) 类 型 的 , Web 容器 会 自动 把 值 的 类 型 转换 成 
属性 的 类 型 ,也 就 是 说 这 种 转换 是 自动 发 生 的 ,而 不 是 由 用 户 通过 程序 来 控制 的 。 这 种 转 
换 如 表 5-2 所 示 。 


表 5-2 自动 转换 属性 的 类 型 


属性 类 型 在 字符 串 值 上 的 转换 
Bean 属性 使 用 setAsText(string-literal) 
boolean 或 Boolean java. lang. Boolean. valueOf(String) 
byte 或 Byte java. lang. Byte. valueOf(String) 
char 或 Character java. lang. String. charAt(0) 
double 或 Double java. lang. Double. valueOf(String) 
int 或 Integer java. lang. Integer. valueOf(String) 
float 或 Float java. lang. Float. valueOf(CString) 
long 或 Long java. lang. Long. valueOf(String) 
short 或 Short java. lang. Short. valueOf (String) 
Object new String(string-literal) 


1. 使 用 表达 式 设置 属性 


在 实际 的 应 用 中 设置 Bean 属性 的 时 候 , 通 常 值 都 是 动态 的 变量 而 非常 量 ,这 就 需要 
使 用 表达 式 或 者 其 他 的 方式 来 给 它们 赋值 。 
关 示 例 5-13 ”使 用 表达 式 设置 StockInfo 的 size 属性 


<jsp:useBean id="stockinfo" class="com.jyw.www.entity.StockInfo"/> 
<jsp:setProperty name="stockinfo" property="size" value=" 
<$=request.getParameter ("size") %$>" /> 
在 示例 5-13 中 使 用 了 表达 式 ,并且 将 它 用 在 setProperty 中 , 它 将 会 接收 从 表单 中 传 
递 过 来 的 数据 ,并 且 用 这 些 值 来 设置 JavaBean 的 属性 。 


2. 使 用 setProperty 的 param 属性 


由 于 request. getParameter( ) 的 返回 值 是 String 类 型 ,我 们 如 果 使 用 request. 
getParameter() 从 表单 中 接收 数据 ,再 将 它 赋 给 JavaBean 中 的 非 String 类 型 的 属性 , 那 
么 就 需要 对 它 进行 转换 。JSP 提供 了 一 个 很 好 的 解决 方法 : 它 允 许 在 使 用 请 求 参数 设置 
JavaBean 属性 的 时 候 , 将 属性 和 参数 相关 联 , 并 自动 执行 从 字符 串 到 数字 、 字 符 和 布尔 类 
型 的 值 的 转换 。 从 示例 5-12 中 对 表单 的 全 部 元 素 与 JavaBean 属性 的 关联 的 演示 ,读者 
应 该 已 经 体会 到 使 用 上 的 方便 性 。 问 题 是 在 某 些 应 用 中 ,只 需要 关联 特定 的 属性 ,这 时 可 
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以 使 用 param 属性 来 指定 输入 参数 ,被 指定 的 请 求 参 数 的 值 会 自动 根据 JavaBean 的 属性 
类 型 转换 ,并 且 将 它 作 为 JavaBean 属性 的 值 。 

示例 5-14 使 用 param 来 将 请 求 参 数 和 JavaBean 属性 关联 ,并 完成 了 示例 5-14 的 
功能 。 

闪 示 例 5-14 ”使 用 param 关联 request 中 的 参数 和 StockInfo 的 属性 


<jsp:setProperty name= "stockinfo" property="price" param="price"/> 
<jsp:setProperty name=" stockinfo " Property= "name" param=" name "/> 


示例 5-14 使 用 param 来 分 别 指定 了 与 JavaBean 的 属性 相关 联 的 请 求 参 数 , 这 样 在 
执行 这 个 JSP 文 件 的 时 候 , 就 会 自动 把 各 自 对 应 的 请 求 参 数 根据 JavaBean 属性 类 型 进行 
转换 后 赋 给 它 。 如 果 指 定 的 请 求 参数 不 存在 ,将 不 采取 任何 的 行动 ,也 就 是 说 ,系统 并 不 
传递 NULL 到 相关 联 的 属性 。 

如 果 请 求 参 数 名 称 和 JavaBean 的 属性 名 称 一 致 ,还 可 以 更 进一步 地 将 param 去 掉 ， 
例如 ,对 于 上 面 的 例子 ,因为 表单 中 的 元 素 名 称 price size 和 JavaBean 的 属性 名 称 一 致 , 
所 以 可 以 将 上 面 的 三 行 修改 如 下 。 

闪 示 例 5-15 ”修改 代码 


<jsp:setProperty name= "stockinfo" property="price" /> 
<jsp:setProperty name=" stockinfo " property= "name" /> 


5.5.3 使 用 jsp:getProperty 获取 JavaBean 的 属性 


要 获取 JavaBean 的 属性 ,可 以 使 用 二 jsp: getProperty 二 动作 标签 。 其 语法 规则 
如 下 : 


<jsp:getProperty name="beanInstanceName" property="propertyName" /> 
闪 示 例 5-16 ”获取 StockInfo 的 color 属性 


<jsp:getProperty name=" stockinfo " property= "color"/> 


它 将 会 自动 调用 StockInfo 的 getColor() 方 法 ,并 且 将 获得 的 返回 值 输出 到 页 面 中 。 


课 后 练习 


1. 结合 以 前 学 过 的 静态 网 页 设计 的 知识 ,以 及 第 1 章 功能 模块 中 设计 的 示意 图 ,为 
“ 佳 衣 屋 ”项 目 设计 美观 而 且 切 合 主题 的 页 面 风 格 , 并 添加 导航 栏 。 
提示 : 参考 第 4 章 中 include 指令 标签 的 用 法 。 
2. 连锁 商业 企业 ,如 深圳 茂 业 百货 、 岁 宝 百 货 等 ,在 其 网 站 上 都 可 以 让 用 户 获 得 各 分 
店 基本 信息 的 功能 。 参 考 这 些 网 站 的 用 户 界面 设计 ,为 “ 佳 衣 屋 ” 项 目 添加 美观 而 且 实 用 
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的 门店 导航 功能 。 

要 求 : 给 出 用 户 选择 分 店 的 接口 , 列 出 用 户 选择 的 门店 的 地 址 、 联 系 方式 等 信息 。 

提示 : 参考 本 章 任务 三 ,完成 对 第 3 章 中 创建 的 storeinfo 数据 表 的 条 件 作 为 分 店名 
的 查询 。 

3. 商品 的 销售 情况 ,是 企业 需要 实时 掌握 的 重要 信息 ,直接 影响 企业 的 营销 .生产 等 
其 他 环节 的 决策 ,请 你 为 " 佳 衣 屋 项目 添加 灵活 的 多 条 件 查询 销售 情况 的 功能 。 

要 求 : 结合 实际 应 用 ,自主 分 析 销 售 情况 的 查询 条 件 都 有 哪些 ?给 出 用 户 条 件 输入 
或 者 选择 的 接口 , 列 出 查询 到 的 销售 情况 列表 。 

提示 : 参考 本 章 任务 四 ,完成 对 第 3 章 中 创建 的 salesinfo 视图 的 多 条 件 查询 。 

4. 本 章 学 习 的 主要 内 容 是 对 数据 表 的 查询 与 显示 ,这 是 中 小 型 Web 项 目 中 必 不 可 
少 的 功能 。 比 如 : 人 事 管理 系统 中 的 员工 信息 的 查询 与 显示 ,客户 关系 管理 系统 中 客户 
信息 的 查询 与 显示 ,图 书 管理 系统 中 书目 的 查询 与 显示 等 。 请 按照 第 1 章 课 后 练习 的 要 
求 , 完 成 项 目 中 相应 的 数据 表 查 询 的 功能 模块 。 
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本 章 学 习 要 点 : 

。 熟练 掌握 DAO 类 中 save() 方 法 的 编写 ; 

。 熟练 掌握 SQL 中 INSERT INTO 语句 的 各 种 用 法 ; 

。 熟练 掌握 如 何在 项 目 中 使 用 第 三 方 组 件 ; 

。 熟练 掌握 使 用 JSPSmartUpload 实现 文件 上 传 的 方法 ; 
。 熟练 掌握 Service 类 的 编写 方法 。 


6.1 任务 一 : 新 商品 信息 的 录入 


问题 : 每 一 季 的 新 品 上 市 前 ,相关 信息 应 该 被 录入 到 数据 库 中 以 备 店员 进行 查询 或 
订购 等 其 他 操作 ,那么 ,使 用 JSP 十 JavaBean 技术 ,如何 向 数据 表 中 添加 一 条 记录 呢 ? 


: 采用 JSP 十 JavaBean 技术 ,完成 向 商品 基本 信息 表 中 添加 新 商品 信息 的 操作 。 
”技能 训练 : 
: 。 DAO 类 的 save 方法 的 编写 规则 。 

。 INSERT INTO 语句 的 语法 规则 。 

。 JSP 中 调用 DAO 类 的 save 方法 。 


6.1.1 编写 DAO 类 中 的 save 方法 


完成 第 5 章 的 学 习 后 ,已 经 为 项 目的 每 个 数据 表 或 者 视图 都 建立 了 相应 的 实体 Bean 
和 DAO 类 ,并 在 DAO 类 中 添加 了 多 个 与 查询 相关 的 方法 。 要 完成 添加 新 商品 信息 的 功 
能 ,需要 在 与 商品 基本 信息 表 对 应 的 DAO 类 GoodsDAO 中 添加 一 个 用 来 向 相应 数据 表 
中 添加 记录 的 方法 ,一 般 将 其 命名 为 save() ,方法 的 声明 示意 如 下 : 


void save(Object obj) 
{0} 


save() 方 法 中 需要 执行 SQL 中 的 INSERT INTO 语句 来 完成 记录 的 添加 ,语法 


中 小 型 Web 项 目 开 发 实战 


如 下 : 

INSERT INTO 表 名 称 ( 列 名 1， 列 名 2, ...)VaLUES ( 值 1, 值 2，.….) 

列 名 的 出 现 顺 序 要 与 值 出 现 的 顺序 相对 应 ,“ 值 1” 应 添加 到 以 “ 列 名 1? 为 名 称 的 列 
中 ,“ 值 2” 应 添加 到 以 “ 列 名 2? 为 名 称 的 列 中 ,其 他 以 此 类 推 。 

下 面 给 出 该 方法 的 完整 源 代码 : 

闪 示 例 6-1 GoodsDAO 类 中 的 save() 方 法 


public void save (Goods goods) { 


a “SS 实体 bean Goods 的 实例 作为 入 口 参数 


InitialContext ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 


Connection conn=ds.getConnection(); 


PreparedStatement pstmt=conn.prepareStatement ("INSERT INTO goods 


(id,name, image,price,color,size) VALUES 
预 编译 (3213,?,?,3,?)",ResultSet.TYPE SCROLL INSENSITIVE, 
a | 和 
INSERT INTO 语 名 | Resultset .CONCUR_READ ONLY) ; 


Pstmt .setString(1，goods .getId()) 
Pstmt .setString(2，goods .getName () ) ; 
Pstmt .setString(3，goods .getImage () ) ; 用 户 输入 的 商品 基本 
Pstmt .setFloat (4, goods.getPrice()); 
Pstmt .setString(5，goods .getColor ()); 
Pstmt .setString(6，goods .getSize())7 


Pstmt .executeUpdate () 
Pstmt .close(); ~ 调用 executeUpdate() 方 法 


} catch (Exception e) { 
// TODO Auto- generated catch block 
e.printstackTrace (); 


信息 替换 SQL 语 
中 的 "?" 占 位 符 


} 


注意 : 这 里 调用 了 PreparedStatement 的 executeUpdate() 方 法 ,第 5 章 的 查询 操作 
都 是 使 用 executeQuery() 方 法 。 


6.1.2 编写 添加 商品 基本 信息 相关 的 JSP 文件 


闪 示 例 6-2 addGoodsForm. jsp 中 用 于 添加 商品 人 库 的 表单 部 分 代码 如 下 : 


<#se@page language="java" contentType="text/html;charset=gb18030"%> 
<form action="addGoods .jsp" method="post"> 
<table align=center> 
<tr> 
<td colspan="2" align= center> 商 品 基本 信息 < /td> 
</tr> 
<tr> 
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<td> 编 码 < /td> 
<td><input type="text" name="id" id="id"></td> 
</tr> 
<tr> 
<td> 名 称 </td> 
<td><input type= "text" name= "name" id="name"></td> 
</Er> 
<tr> 
<td> 销 售 指导 价 </td> 
<td><input type="text" name=" 
</tr> 
<tr> 
<td> 颜 色 < /td> 


<td><input type="text" name="color" id="color"> 


price" id="price"></td> 


<input type="hidden" name="size" value= "S-XL"></td> 
</tr> 


<tr> 
<td colspan=2 align=center> 


<input type="submit" value= "添加 "><input type="reset" value= 
" 重 置 "></td> 
</tr> 
</table> 
</form> 


在 浏览 器 中 ,addGoodsForm. jsp 程序 的 执行 效果 如 图 6-1 所 示 。 


图 6-1 商品 人 库 基本 信息 填写 表单 示意 


输入 商品 基本 信息 后 , 单 击 图 6-1 的 “添加 "按钮, 跳 转 到 addGoods. jsp 页 面 处 理 , 其 
源 代码 如 下 。 


2X 示 例 6-3 addGoods. jsp 


<%@page language="java" import="java.util.*" contentType="text/html;charset= 


GB18030"%> 

<% 
request .setCharacterEncoding ("GB18030"); 设置 输入 /输出 编码 方式 
response.setContentType ("text/html;charset=GB18030") 

%> 


hia 
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<jsp:useBean id="goods" class="com.jyw.www.entity.Goods" scope="page" /> 


<jsp:setProperty name="goods" property="*" /> 一 一 i ta 


<jsp:useBean id="goodsDAO" class="com.jyw.www.dao.GoodsDAO" scope="page" /> 


<% 调用 GoodsDAO 的 save 方 法 完成 
goodsDAO.save (goods) 一 向 数据 表 中 插入 一 条 新 记录 的 操作 


response.sendRedirect ("showGoods.jsp"); 


跳 转 到 商品 信息 显示 页 


各 > 


以 上 代码 执行 完毕 后 ,可 以 看 到 goods 数据 表 添 加 了 一 条 新 记录 。 
6.2 任务 二 : 商品 图 片 的 上 传 


完成 任务 一 后 ,商品 基本 信息 并 没有 录入 完整 ,示例 6-1 中 的 INSERT INTO 语句 中 
有 6 个 占 位 符 , 但 从 示例 6-2 的 addGoodsForm. jsp 的 表单 中 只 接收 到 5 个 参数 ,还 缺少 
商品 的 展示 图 片 的 文件 名 没有 添加 到 数据 库 中 ,图 片 文件 也 没有 上 传 至 系统 中 , 接 下 来 将 
这 一 功能 补充 完整 。 


”将 商品 的 展示 图 片上 传 至 项 目 根 目 录 下 的 images 子 目 录 下 ,要 求 每 个 图 片 大 小 ， 
”不 能 超过 200KB, 并 将 上 传 图 片 的 名 称 添加 到 goods 数据 表 中 。 
”技能 训练 : 
: 。 文件 上 传 元 素 的 使 用 。 

。 含有 文件 上 传 元 素 的 表单 的 属性 设置 。 

。 第 三 方 组 件 的 使 用 。 

。 使 用 JSPSmartUpload 完成 图 片 的 上 传 。 

。 Service 类 的 编写 。 


6.2.1 文件 上 载 组 件 的 使 用 


首先 在 addGoodsForm. jsp 表单 中 的 商品 名称” 和 ”销售 指导 价 ? 之 间 添 加 一 个 文件 
上 载 组 件 。 


<tr> 


ds 文件 上 载 组 件 的 type 属 性 为 file 


<td><input type="file" name="file" id="file"></td> 


</tr> 
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表单 中 enctype 属性 定义 了 表单 的 MIME 编码 方式 ,默认 情况 下 这 个 编码 格式 是 
application/x-www-form-urlencoded, 不 能 用 于 文件 上 传 ; 只 有 使 用 了 multipart/form- 
data 编码 方式 时 ,才能 传递 文件 数据 ,所 以 含有 文件 上 载 组 件 的 表单 应 设置 为 enctype 一 
"multipart/form-data" ,表单 的 method 应 设置 为 "POST”。 将 示例 6-2 中 的 addGoodsForm 
.jsp 的 表单 属性 设置 做 如 下 修改 : 


<%@page language="java" contentType="text/html;charset=gb18030"%> 
<formaction="addGoods.jsp" method="post" enctype="multipart/form-data"> 


带 文 件 上 传 功能 的 表单 的 MIME 编 码 
表单 的 传输 方式 必须 是 post ™ 方式 必须 是 multipart/ form-data 


重新 执行 addGoodsForm. jsp, 其 在 浏览 器 中 的 显示 效果 如 图 6-2 所 示 。 
http. /flocalhost 6000/add000dsForn, jsp 国 [EIEES 


商品 基本 信息 


单 击 “ 浏 览 " 按 钮 可 
打开 资源 浏览 器 ， 选 择 
需要 上 传 的 文件 


图 6-2 添加 了 文件 上 载 组 件 后 的 商品 入 库 基 本 信息 填写 页 面 


6.2.2 第 三 方 组 件 JSPSmartUpload 的 使 用 


在 项 目的 开发 过 程 中 常常 需要 使 用 第 三 方 组 件 来 提高 开发 效率 。 第 三 方 组 件 就 是 由 
Oracle 以 外 的 公司 开发 且 已 经 封装 好 某 些 既定 功能 的 组 件 ,只 需要 正确 地 进行 使 用 , 便 
可 实现 这 些 功 能 。jspSmartUpload 就 是 一 个 开源 的 文件 上 传 .下 载 组 件 ,通过 使 用 
jspSmartUpload, 可 以 很 方便 地 实现 文件 的 上 传 ,并 设置 上 传 文件 的 存放 目录 和 限制 上 传 
文件 的 大 小 和 类 型 。 要 使 用 jspSmartUpload 组 件 .需要 下 载 jspSmartUpload 的 jar 文件 
(请 读者 自行 下 载 )。 下 载 完成 后 将 jar 文件 放 入 到 项 目 根 目录 下 的 Web-INF/lib 子 目录 
下 。 为 了 使 项 目 源 代码 能 够 编译 通过 ,还 要 将 其 添加 到 项 目的 Build Path 中 。 使 用 
MyEclipse 向 项 目的 Build Path 中 添加 jar 文件 的 方法 如 图 6-3 一 图 6-6 所 示 。 

下 面 编写 FileManager. java, 其 uploadFile 方法 中 使 用 了 jspSmartUpload 的 方法 来 
实现 文件 上 传 。 可 将 FileManager. java 文件 创建 到 项 目的 util 包 中 。 

闪 示 例 6-4 FileManager. java 


| 引 入 javax.servletjsp.PageContext 类 | 
Package com.jyw.www.util; 


import javax.servlet.jsp.PageContext; 


import com.jspsmart .upload.* 天 -一 引入 com.jspsmart.upload 包 


Lis. 
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图 6-3 MyEclipse 向 项 目的 Build Path 中 添加 jar 文件 


(D) 单 击 左 侧 菜单 中 的 (2) 单 击 Libraries 选 项 卡 。 (3) 单 击 Libraries 选 项 卡 
Java Build Path 右 侧 的 AddJARs 按 钮 


pe ler te Java Build path 


Resource | 
[8 soarer [SB majeets] Bh hreie [全 Order ad Exper' 
ThRs and class folders on the build path 
Bh Jers HE 5 Librarier 


-Wh JIE Systen Librury [Sun JRE 1.5.0.11] 


Javadoe Location 
WW MyEclipse 
Project Refereness 
Befactoring History 
Bun/Debue Settings 


Wow Jr 


图 6-4 打开 Libraries 选项 卡 


Ehoose the archives to be added to the build path' 


jt ip 
日 匡 jy 
日 多 YebRoot 
BB YEB-ITIF 
(DD 选中 WebRooWeb-INF/lib | “一 晤 IE 
目录 下 的 jar 文 件 
人) 单 击 OK 按钮 


图 6-5 选择 需要 添加 的 jar 文件 
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可 以 看 到 jar 文 件 已 经 
出 现在 libraries 选 项 卡 


车 


Java Build Path 


ster = , 
Builders BB Somes| BB Projects| BW Litreies [他 
Ws er IARs and class folders on the build path 
ve Cede Style 
E: SE jmteom zh CW jer ~ jyw/YebRoo /ED-TIN /lib 
-Bh Javs EE 5 Libraries 
-Bh JRE System Library [Sun JEE 1.5.0_11] 


Project References 
Refactoring History 


Run/Debus Settings 


单 击 OK 按钮 
图 6-6 完成 jar 文 件 的 添加 
import com. jyw.www.entity.*; 
public class FileManager { 


Public void uploadFile (PageContext pageContext) 
! 


SmartUpload smartupload=new SmartUpload (); 
创建 SmartUpload 实 例 
try { 


smartupload.initialize (pageContext) = 初始 化 


smartupload.setTotalMaxFileSize(200000); 
设置 上 的 总 的 大 小 上 限 为 200KB 
Smartupload.upload () 和 上 传 文件 设 传 文件 的 总 的 大 小 上 限 为 


smartupload.save ("/images") ;之 设置 上 传 文件 存放 目录 为 /images 
} catch (Exception e) { 


e.printStackTrace () 


6.2.3 编写 GoodsService 类 


要 完成 本 任务 ,涉及 两 个 “动作 ”, 一 是 要 完成 文件 的 上 传 ;二 是 向 数据 表 goods 中 添 
加 一 条 新 记录 ,这 两 个 “动作 ”分 别 由 示例 6-4 中 FileManager 类 的 uploadFile() 方 法 和 示 
lz 
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例 6-1 中 GoodsDAO 中 的 save() 方 法 完成 。 对 于 这 种 较为 复杂 的 业务 逻辑 ,需要 为 项 目 
引入 一 个 业务 层 (service) 来 处 理 。 
为 项 目 添 加 com. jyw. www. service 包 , 在 包 中 新 建 GoodsService 类 , 源 代码 如 下 。 


闪 示 例 6-5 GoodsService. java 
package com.jyw.www.service; 


import javax.servlet .jsp.PageContext; 
import com.jyw.www.dao.GoodsDAO; 
import com.jyw.www.entity.Goods; 
import com.jyw.www.util.FileManager; 
import com.jspsmart .upload.*;} 


public class GoodsService { 
public void save (PageContext pageContext) throws Exception 


{ 上 传 文件 


SmartUpload smartupload=new FileManager () .uploadFile (PageContext) 

Goods goods=new Goods () 

File file=smartupload.getFiles() .getFile(0); 

goods.setId(smartupload.getRequest () .getParameter ("id")); 

获取 通 goods .setName (smartupload.getRequest () .getParameter ("name")); 

goods .setPrice (Float.parseFloat(smartupload.getRequest().getParameter 

递 的 商 ("price"))); 

品 信息 | | goods .setColor (smartupload.getRequest () .getParameter ("color")); 
goods .setSize (smartupload.getRequest () .getParameter ("size")); 
goods.setImage (file.getFileName ()); 


new GoodsDAO () .save (goods) ; 一 一 向 数据 表 中 插入 新 记录 


} 


表单 中 设置 enctype 二 "multipart/form-data" 后 ,除了 file 类 型 组 件 外 ,其 他 元 素 的 
value 通过 request. getParameter 都 不 能 取得 ,在 示例 6-5 中 ,使 用 了 JSPSmartUpload 组 
件 来 解决 这 一 问题 ,使 用 了 smartupload. getRequest(). getParameter("xxxx") 来 获取 表 
单元 素 的 值 。 

在 添加 了 Service 层 后 ,addGoods. jsp 也 需要 做 相应 的 修改 ,修改 后 代码 如 下 : 

闪 示 例 6-6 ”修改 后 的 addGoods. jsp 


<%@page language="java" import="com.jyw.www.service.* " 
contentType="text/html;charset=gb18030"%> 


< 
GoodsService gs=new GoodsService(); 
gs.save (pageContext) :== 二 = 调用 GoodsService 类 的 save() 方 法 
response.sendRedirect ("showGoods.jsp"); 

多 > 


至 此 ,文件 将 被 上 传 到 /images 目录 下 ,文件 名 存放 在 数据 表 goods 中 。 
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6.3 任务 三 : 分 店 批量 申请 进货 


新 货 上 市 时 ,分 店 需 要 从 总 店 进货 ,这 一 过 程 由 分 店 向 总 店 发 起 “订单 申请 ”开始 , 即 
向 存储 订单 信息 的 orders 数据 表 中 添加 新 记录 ,所 以 也 是 用 数据 表 的 插入 操作 来 完成 。 
人 
低 ,用 户 体验 不 理想 。 下 面 与 读者 探讨 如 何 * 批 量 申请 进货 ” 


“任务 目标 E 
; 分 店 店员 进入 进货 页 ,可 看 到 所 有 总 店 库存 商品 的 列表 , 勾 选 需要 进货 的 条 目 ， 并 | 
; 在 其 后 输入 进货 量 , 单 击 "提交 "按钮 后 ,完成 批量 进货 的 申请 。 
技能 训练 ， 
MySQL 批量 添加 记录 语句 的 使 用 方法 。 
批量 进货 申请 功能 的 底层 实现 是 向 orders 表 中 添加 多 条 记录 ,这 当然 可 以 通过 多 次 
调用 INSERT INTO 语句 来 完成 ,但 这 样 会 增加 服务 器 的 负荷 ,因为 每 执行 一 次 SQL 语 
句 ,服务 器 都 要 对 SQL 语句 进行 分 析 、 优 化 等 操作 。 
MySQL 提供 了 另 一 种 解决 方案 ,即使 用 一 条 INSERT INTO 语句 来 插入 多 条 记录 。 
INSERT INTO table (columl, colum2,...,column) VALUES 


(valuell,valuel12, ...valueln), 
(value21,value22, ...value2n), 


(valuenl,valuen2, ...valuenn) 


注意 : 上 述 批量 插入 的 INSERT INTO 语句 不 是 标准 的 SQL 语句, 仅 对 MySQL 数 
据 库 的 访问 有 效 。 


6.3.1 重 载 OrdersDAO 类 中 的 save 方法 


在 完成 本 章 任务 一 的 过 程 中 ,为 GoodsDAO 类 添加 了 一 个 用 来 插入 一 条 记录 的 save() 
方法 ,可 以 看 做 是 每 个 DAO 类 的 “标准 配置 "。 与 存放 订单 信息 的 orders 数据 表 相 对 应 
的 DAO 类 OrdersDAO 类 也 应 有 一 个 save(Orders orders) 方 法 ,用 来 向 orders 表 中 插入 
一 条 “订单 ”记录 。 读 者 可 参照 任务 一 的 步骤 自行 完成 。 

但 任务 二 的 要 求 是 一 次 插入 多 条 记录 。 下 面 在 OrdersDAO 类 中 重 载 save() 方 法 。 
由 于 入 口 参数 为 批量 订单 信息 ,所 以 使 用 了 ArrayList 类 型 来 作为 入 口 参 数 类 型 , 源 代 码 
如 下 。 

闪 示 例 6-7 OrdersDAO 中 的 save(ArrayList<Orders 二 orderslist) 方 法 


public void save (ArrayList<Orders>orderslist) { 
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InitialContext ctx; 
省 
ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds.getConnection(); 
Statement stmt=conn.createSstatement (); 
java.util.Date ud=new java.util .Date(); 
java.sql.Timestamp stp=new java.sql.Timestamp (ud.getTime ()); 
String sgl="INSERT INTO orders 
(goodsid, size,quantity,storenum,applydate,orderstatus) VALUES"; 


一 次 插入 多 条 记录 

的 INSERT INTO 有 昌 A jy 

lS 因为 订货 条 目 数 不 确定 ， 所 以 使 用 循环 

语句 的 前 半 部 分 2 来 遍历 orderslist， 并 将 订货 信息 逐条 
for (Orders order : orderslist) 加 入 INSERT INTO 语 句 的 VALUES 部 分 


sql=sql+" ('"+torder.getGoodsid()+"','"+order.getSize() 
+"',"t+order.getQuantity()+","+order.getStorenum() 
+ tstpt" "+"'! 待 处 理 '+") n+ 


sql=sql.substring(0, sql.length()-1); 


stmt .executeUpdate (sql); 去 掉 INSERT INTO 语 句 
} catch (Exception e) { 末尾 的 “,"， 使 得 语法 正确 


e.printstackTrace (); 


6.3.2 编写 与 添加 批量 订单 相关 的 JSP 文件 


下 面 编写 分 店 进行 批量 进货 的 JSP 页 面 ,该 页 面 列 出 了 总 店 所 有 库存 商品 的 信息 ， 
可 供 分 店 进行 进货 选择 ,并 可 输入 订货 量 。 
闪 示 例 6-8 orderForm.jsp 


<%Q@page language="java" import="java.util.* ,com.jyw.www.dao.*, 
com.jyw.www.entity. * "contentType="text/html;charset=gb18030"%> 
<HTML> 
<BODY> 
<CENTER> 
总 店 库存 商品 列表 
<FORM ACTION= "ordersSubmit .jsp" method="post" name="form2"> 
< TABLE border=1> 
<TR> 
<TD align=center><B> 选 择 </B>< /TD> 
<TD align=center><B> 图 片 </B></TD> 
<TD align=center><B> 商 品 编号 </B>< /TD> 


120 


第 6 章 商品 入 库 


<TD align=center><B> 尺 码 </B>< /TD> 
<TD align=center><B> 单 价 </B></TD> 
<TD align=center><B> 库 存量 </B></TD> 
<TD align=center><B> 进 货 量 </B></TD> 
</TR> 
<% 
StockInfoDAO stockinfodao=new StockInfoDAO(); 
List<StockInfo>stocklist=stockinfodao.findByStoreName (" 佳 衣 屋 总 店 华强 北 店 "); 


for (StockInfo stockinfo : stocklist) { 
从 StockInfo 视 图 中 查找 “ 佳 衣 屋 


> 总 店 华强 北 店 "的 库存 信息 
<TR ALIGN= CENTER> 


<TD><input type="checkbox"name="goodsid" value="< 多 Stockinfo.getId() 和 "> 


Se 采用 复 选 框 ， 使 得 分 店 店员 可 一 次 选择 多 条 商品 


<TD><img width=60 height=60 src="/images/<%=stockinfo.getImage ()%> "> 
</TD> 

<TD><%=stockinfo.getId()®%></TD> 
<TD><%=stockinfo.getSize()®%></TD> 
<TD><%=stockinfo.getPrice()®%></TD> 
<TD><%=stockinfo.getstock()%></TD> 

<TD><input type="text"name="<$=stockinfo.getId()%>quan"size=6></TD> 


RAs “进货 数量 "文本 框 的 名 字 中 包含 了 该 条 "商品 编码 "， 
以 便于 处 理 商 品 与 进货 数量 的 对 应 关系 


<input type="hidden" value="<%=stockinfo.getSize() %>" 
name="<%=stockinfo.getId() $>size"> 


< 不 需要 用 户 干涉 的 参数 
} | 用 “隐藏 "类 型 的 元 素来 传递 


多 > 

</TABLE> 
<input type="hidden" value="1l" name="storenum"> 

分 店 默 认为 “一 分 店 ”。 在 第 9 章 中 我 们 将 学 习 如 何 
自动 识别 当前 用 户 所 属 分 店 ， 再 将 此 功能 进一步 完善 

<input type="submit" value= "确定 进货 "> 

< /FORM> 

</CENTER> 

</BODY> 

</HTML> 


该 页 的 执行 效果 如 图 6-7 所 示 。 

勾 选 需要 进货 的 商品 ,并 填写 进货 量 后 , 单 击 * 确 定 进货 按钮 ,表单 数据 将 提交 给 
ordersSubmit. jsp 文件 来 处 理 ,该 文件 的 源 代码 如 示例 6-9 所 示 。 

闪 示 例 6-9 ordersSubmit.jsp 


<%@page language="java" 
import="java.util.* ,com.jyw.www.dao. * ,com.jyw.www.entity.*™" 
contentType="text/html;charset=gb18030"%> 

<% 
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画 吾 画 | 
GO ~ [其 ntpyocahostaoso/ordersformjsp 司 好 |[x] [Eee , 你 就 知道 1P| 


窗 窒 名 http//localhost8080/ordersFormj.. 全 " 国 7 了 慷 ” 且 Pp)v 闪 IRO)>” 


总 店 库存 商品 列表 
商品 编号 尺码 | 单价 


JYW12CKB0| S 198.0 | 78 


JYW12CKN1 123.0| 88 


JYW12GZY1 99.0 | 15 


JYW12CKY2 158.0| 30 


JYWI2YWY1 | M |50.0| 50 


可 
[| [| |@ memet| ina: EA [R100% > 用 


图 6-7 用 户 批量 订货 页 面 的 显示 效果 


OrdersDAO ordersDao=new OrdersDAO () 7 
String[] goodsid=request.getParameterValues ("goodsid"); 
if (goodsid !=null) { 
ArrayList<Orders>orderlist=new ArrayList<Orders> (); 
int storenum=Integer.parseInt (request .getParameter ("storenum")); 
for (int i=0; i<goodsid.length; i++) { 
Orders order=new Orders (); 
order.setGoodsid(goodsid[i]); 
order.setSize (request .getParameter (goodsid[i]+"size")); 
order.setQuantity(Integer.parseInt (request .getParameter 
(goodsid[i]+"quan"))); 
order.setStorenum(storenum); 
orderlist.add (order); 
} 
ordersDao.save (orderlist); 
response.sendRedirect ("orderSubmitSuccess.jsp"); 
} else { 
response.sendRedirect ("orderSubmitFail.jsp"); 


和 > 


单 击 “ 提 交 ” 按 钮 后 ,将 完成 订单 的 提交 .图 6-8 为 完成 订货 记录 添加 后 orders 数据 表 
中 的 记录 ,其 中 id 为 8 和 9 的 记录 为 一 次 添加 的 两 条 新 订货 信息 。 


i22 


历 限制 和 第 - 行 ， 4 二 和 和: 区 


2 2013-02-22 15:26:51| 
3|rrw1267Y1 a0] 2 2013-02-22 15:34:26| 2013-02-22 15:37:26| 己 发 货 
lzcitao a ls 2013-03-31 21:00:18| 0LL) | 竺 处 理 
lyywi2ciar 2z| i | 2013-03-31 21:00:18 (00LL) | 竺 处 理 
CC [CE 天 | [7 [EC 
:jyw 表单 : orders 


图 6-8 orders 数据 表 中 出 现 新 记录 


6.4 知识 扩展 


6.4.1 使 用 INSERT INTO 语句 插入 记录 的 其 他 用 法 


在 完成 本 章 任务 的 过 程 中 ,使 用 了 SQL 的 INSERT INTO 语句 向 数据 表格 中 插入 
新 记录 。INSERT INTO 语句 还 可 以 根据 不 同 的 应 用 情况 ,采用 不 用 的 语法 格式 。 

本 章 第 一 节 中 给 出 INSERT INTO 语句 ,采用 了 列 名 称 来 指出 需要 写 人 的 列 ,这 样 
的 设计 使 程序 有 较 好 的 可 读 性 。 为 了 提高 执行 效率 , 列 名 也 可 以 由 列 序号 来 代替 ,语法 
如 下 : 


INSERT INTO 表 名 称 ( 列 1 序号 , 列 2 序 号, ...) VALUES ( 值 1, 值 2,..…) 
采用 上 述 语法 ,示例 6-1 中 的 INSERT INTO 预 编 译 语 句 可 做 如 下 改写 : 


PreparedStatement pstmt= conn .prepareStatement ("INSERT INTO goods (1,2,3,4,5,6) 
VALUES(?,?,?,?,?,?)",ResultSet .TYPE SCROLL_ 
INSENSITIVE, ResultSet .CONCUR READ ONLY); 


如 果 新 插入 的 记录 的 每 一 列 都 要 写 入 , 即 值 的 个 数 与 表 的 列 数 相同 ,还 可 以 省 略 列 
名 / 列 序号 部 分 ,语法 如 下 : 
INSERT INTO 表 名 称 VALUES ( 值 1, 值 2,.…) 


采用 上 述 语 法 ,示例 6-1 中 的 INSERT INTO 预 编 译 语句 可 做 如 下 改写 : 


PreparedStatement pstmt= conn.prepareStatement ("INSERT INTO goods VALUES 
(3,23,23,?3,?,?3)",ResultSet.TYPE SCROLL INSENSITIVE, 
ResultSet .CONCUR READ ONLY); 


6.4.2 使 用 INSERT INTO 语句 进行 表 复 制 


开发 Web 应 用 程序 的 过 程 中 ,会 遇 到 需要 将 一 个 数据 表 的 全 部 或 者 部 分 数据 复制 到 
另外 一 个 数据 表 中 。 
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如 果 两 个 数据 表 的 列 一 致 ,并 且 需 要 复制 全 部 数据 ,可 以 采用 下 述 语句 : 
INSERT INTO 目标 表 SELECT * FROM 来 源 表 


如 果 两 个 数据 表 的 列 一 致 ,但 仅 需要 复制 满足 某 种 条 件 的 全 部 数据 ,可 以 采用 下 述 
语法 : 


INSERT INTO 目标 表 SELECT x FROM 来 源 表 WHERE 条 件 

如 果 只 需要 复制 指定 的 列 , 可 以 采用 下 述 语句 : 

INSERT INTO 目标 表 ( 列 1, 列 2, …) SELECT 列 1, 列 2, .FROM 来源 表 WHERE 条 件 
注意 : 列 的 顺序 必须 一 致 。 

为 了 避免 复制 后 出 现 重复 的 记录 ,可 以 采用 下 述 语句 : 


INSERT INTO 目标 表 ( 列 1, 列 2, ...) SELECT 列 1, 列 2, ...FROM 来 源 表 
WHERE NOT EXSIT (selectx from 目标 表 WHERE 条 件 ); 


读者 应 根据 应 用 的 实际 情况 优化 SQL 语句 的 使 用 。 
6.4.3 executeQuery 和 executeUpdate 方法 的 比较 


在 第 5 章 中 ,执行 SQL SELECT 语句 使 用 了 Statement/PrepareStatement 接口 的 
executeQuery() 方 法 ,这 个 方法 用 于 执行 SELECT 语句 ,产生 单个 结果 集 。 

本 章 中 执行 SQL INSERT INTO 语句 使 用 了 executeUpdate() 方 法 ,这 一 方法 用 于 
执行 INSERT INTO 以 及 后 续 章 节 中 的 UPDATE 或 DELETE 语句 以 及 SQL DDL( 数 
据 定义 语言 ) 语 句 , 如 CREATE TABLE 和 DROP TABLE。executeUpdate() 方 法 的 返 
回 值 类 型 为 整 型 , 值 为 语句 执行 后 受到 影响 的 行 数 。 

读者 可 以 这 样 理解 和 记忆 : 语句 执行 后 ,数据 表 内 容 不 变 , 则 使 用 executeQuery()， 
语句 执行 后 ,数据 表 的 内 容 有 更 新 , 则 使 用 executeUpdate()。 


课 后 练习 


1. 为 “ 佳 衣 屋 ”项 目 实现 “添加 新 用 户 ” 的 功能 。 

要 求 : 给 出 为 系统 添加 新 用 户 的 UI 页面 ,在 该 页 中 可 以 输入 用 户 的 用 户 名 、 密 码 、 真 
实 姓名 ,选择 该 用 户 所 属 分 店 ,选择 该 用 户 的 权限 (管理 员 / 总 店 店员 /分 店 店员 )。 单 击 
“提交 ”按钮 后 ,将 用 户 名 、 密 码 、 真 实 姓名 .选择 所 属 分 店 和 使 用 权限 添加 到 第 3 章 创 建 的 
employee 数据 表 中 。 

提示 : 参考 本 章 任务 一 ,完成 向 第 3 章 创 建 的 employee 数据 表 中 插入 一 条 新 记录 。 

2. 为 stock 库存 信息 表 的 数据 库 操作 的 StockDAO 类 添加 save 方法 。 

要 求 : 为 StockDAO 类 添加 save() 方 法 ,该 方法 完成 向 第 3 章 中 设计 的 stock 数据 表 
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中 添加 一 条 记录 的 功能 。 

提示 : 参考 本 章 示例 6-1 ,为 StockDAO 类 添加 save() 方 法 。 

3. 本 章 主要 学 习 了 如 何 向 数据 表 中 的 插入 新 记录 ,这 也 是 中 小 型 Web 项 目 中 必 不 
可 少 的 功能 。 比 如 : 人 事 管理 系统 中 的 员工 信息 (批量 ) 的 录入 ,客户 关系 管理 系统 中 客 
户 信 息 ( 批 量 ) 的 录入 ,图 书 管理 系统 中 书目 信息 的 (批量 ) 的 录入 等 。 请 结合 第 1 章 课 后 
练习 的 要 求 , 完 成 项 目 中 相应 的 向 数据 表 插 入 新 记录 的 功能 模块 。 


I29. 


第 7 章 商品 信息 的 修改 和 删除 


本 章 学 习 要 点 : 

。 熟练 掌握 DAO 类 中 update() 方 法 的 编写 方法 ; 
。 熟练 掌握 通过 URL 传递 参数 ; 

。 熟练 掌握 通过 表单 隐藏 元 素 传递 参数 ; 

。 熟练 掌握 DAO 类 中 delete() 方 法 的 编写 方法 。 


问题 : 第 5 章 中 为 系统 添加 了 新 商品 的 功能 。 商 品 信息 在 人 工 录 入 时 很 可 能 出 现 错 
误 , 出 现 错误 后 系统 要 提供 修改 的 功能 ,以 便 更 正 。 这 个 功能 是 如 何 实现 的 呢 ? 


7.1 任务 一 : 商品 基本 信息 的 修改 


; 任务 目标 : : 
| 添加 “修改 ”按钮 , 当 单 击 “ 修 改 ” 按 钮 或 超 链 接 时 进入 修改 页 面 ,该 条 商品 的 原 有 
”信息 出 现在 相应 元 素 中 ,用 户 重新 输入 需要 修改 的 信息 后 单 击 “ 提 交 ” 按 钮 ,提交 修改 
; 的 信息 。 
”技能 训练 : 
| 。DAO 类 中 update() 方 法 的 编写 规则 。 

。 JSP 中 通过 URL 或 隐藏 元 素来 传递 参数 。 


7.1.1 编写 DAO 类 中 的 update 方法 


为 了 完成 goods 数据 表 中 记录 信息 的 修改 ,首先 要 为 GoodsDAO 类 添加 update( ) 方 


法 ,在 此 方法 中 需要 用 到 SQL UPDATE 语句 来 指明 需要 修改 满足 什么 条 件 的 记录 ,修改 
的 是 哪些 列 , 改 为 何 值 , 其 语法 规则 是 : 


UPDATE 表 名 称 SET 列 名 称 = 新 值 WHERE 列 名 称 = 某 值 
闪 示 例 7-1 GoodsDAO 类 中 的 update(Goods goods) 方 法 


public void update (Goods goods) throws Exception { 
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InitialContext ctx; 
trYy { 
ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds.getConnection(); 
PreparedStatement pstmt=conn.prepareStatement (" UPDATE goods SET color=?， 
name=?, size=?, price=?,image=?WHERE id=? "); 


Pstmt .setString(1，goods .getColor ()); 
预 编译 UPDATE 语 名 


Pstmt .setString(2，goods .getName () ) 7 
Pstmt .setString(3，goods .getSize()) 7 
Pstmt .setFloat (4，goods .getPrice ()) 7 
Pstmt .setSttring(5，goods .getImage () ) 7 
pstmt.setstring(6, goods.getId()); 
pstmt .executeUpdate (); SSYHRFUPDATE 语 铝 ] 
} catch (Exception e) { 
e.printStackTrace () 
} 


7.1.2 在 显示 商品 信息 页 添加 进入 修改 信息 页 的 用 户 入 口 


在 第 5 章 完成 任务 一 的 过 程 中 ,编写 了 showGoods. jsp, 用 于 显示 商品 的 基本 信息 。 
要 完成 本 章 的 任务 一 ,需要 在 showGoods. jsp 中 为 用 户 添加 一 个 “功能 性 入 口 ”以便 将 
用 户 导 向 商品 信息 修改 页 ,同时 传递 UPDATE 语句 所 需 的 条 件 部 分 ,指明 哪 一 条 商品 的 
信息 需要 修改 。 

通常 有 两 种 形式 实现 这 一 功能 ,在 浏览 器 中 分 别 表 现 为 超 链 接 和 按钮 。 

形式 一 ” 超 链接 

采用 超 链 接 形式 的 功能 入 口 ,要 配合 以 URL 传递 参数 ,格式 如 下 : 


xxx.jsp ?paraml=valuel & param2=value2 &...& paramN=valueN 
闪 示 例 7-2 添加 了 进入 修改 商品 信息 页 超 链 接 后 的 showGoods. jsp 


<%@page contentType="text/html; charset=gb18030" language="java"®> 
<%@page import="com.jyw.www.entity.* ,com.jyw.www.dao. * ,java.util.*x*"%> 
<HTML> 
<BODY> 
<CENTER> 
<TABLE border=1> 
<% 
GoodsDAO goodsDao=new GoodsDAO (); 
ArrayList<Goods>goodsList=goodsDao.findAll (); 
Iterator<Goods>it=goodsList.iterator(); 
while (it.hasNext ()) { 


<TR ALIGN=CENTER> 
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for (int j=0; j<4; j++) { 
Goods goods= (Goods) it.next (); 


<TD> 
<TABLE border=1> 
<TR> 
<TD colspan="4"> 
<img src="/images/<%=goods.getImage()®%>" height=220 
</TD> 
</TR> 
<TR><TD colspan="3"><%=goods.getId()%></TD></TR> 
<TR><TD colspan="3"><%=goods.getName ()%></TD></TR> 
<TR><TD><%=goods.getColor()®%></TD> 
<TD><%=goods.getSize()%></TD> 
<TD><%=goods.getPrice()%>Y¥</TD> 
</TR> 
<TR><TD colspan="3"> 


width=180> 


<a href=updateGoodsForm.jsp?id=<%=goods.getId()%>> 修 改 </a> 


</TD> 


</TR> 添加 超 链接 形式 的 “用 户 功能 性 
< /TABLE> 入 口 "， 使 用 URL 传 递 参 数 id 


</TD> 


<% 
if (!it.hasNext ()) 
break; 


%> 

</TABLE> 

< /CENTER> 

</BODY> 

</HTML> 

执行 效果 如 图 7-1 所 示 。 

形式 二 ”按钮 

采用 按钮 形式 的 功能 入 口 , 同 时 传递 参数 ,又 有 两 种 方式 : 
。 使 用 表单 hidden 元 素 

。 使 用 button 元 素 

(1) 使 用 表单 hidden 元 素 传递 参数 


<a href=updateGoodsForm.jsp?id=<%=goods.getId()$>> 修 改 </a> 
将 上 面 的 语句 更 改 为 下 面 的 代码 即 可 。 


<form action="updateGoodsForm.jsp"> 
<input type="hidden" name="id" value="<%®%=goods.getId() 


%>"> 


SO 


帘 次 熙 httpV/localhosts0s0/showGoodsjsp 


习 回 因 


偷 ~ 园 "竹马 FE 全 IRO YY” 


[下 . sens 


可 


JYW12QKB2 JYW12CKBO JYW12CKN1 JYW12CKY2 
牛仔 裤 牛仔 裤 牛仔 圈 牛仔 裤 
监 [S-XL|¥81.0 | 川 蓝 [S-XL[¥98.0 | 川 水 洗 黑 [Ss-xL[¥123. 0|| 蓉 8-xL|¥¥158.0 
修改 修改 修改 修改 = 
3 [TT [ TT | fmemetl me [区 oo 
图 7-1 带 超 链接 “修改 ”功能 入 口 的 商品 信息 显示 页 


<input type="submit" value=" 修 改 "> 


</form> 


(2) 使 用 button 元 素 传 递 参数 


<a href=updateGoodsForm.jsp?id=<%=goods.getId()%>> 修 改 </a> 


将 源 文件 中 如 上 的 语句 更 改 为 下 面 的 代码 即 可 。 


input type= "button" name= "button" value=" 修 改 " onClick="document .1location. 
href=updateGoodsForm.jsp?id=<%=goods.getId()%>"/ 


采用 按钮 形式 的 功能 入 口 的 页 面 执行 效果 如 图 7-2 所 示 。 


BO |B htp/ocalhostaos0/showso0dsjsp 
帘 窗 镍 htp//localhosta080/showGoodsjsp 


习 图 网 [Eee 下 .aa 


EE 


区 ~ 园 - 巾 ~ 四 mp 和 IROv” 


图 
| 
| 曾 
| 而 
| 昌 
JYW12QKB2 JYW12CKBO JYWI2CKEN1 | JYW12CKY2 
牛仔裤 和 牛仔裤 牛仔 祥 牛仔 裤 
蓝 [S-XL|¥81.0 “| 蓝 [S-XL| 入 98.0 | 水 洗 黑 [S-XL| 计 123. 0|| 洪 -XL| 主 158.0 
| | | | | 
国 
区 [TT Ti | lmernel Snest ee he 


图 7-2 带 按钮 “修改 ”功能 入 口 的 商品 信息 显示 页 
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7.1.3 编写 修改 商品 信息 相关 的 JSP 页 面 


7.1.2 小 节 中 无 论 是 单 击 超 链接 还 是 按钮 都 将 转向 updateGoodsForm. jsp, 并 传递 
了 指定 商品 的 货品 编号 作为 参数 。 下 面 编写 这 一 JSP 文件 。 
闪 示 例 7-3 修改 商品 信息 的 JSP 页 面 updateGoodsForm. jsp 


<se@page language="java" 
import="java.util.* ,com.jyw.www.dao.* ,com.jyw.www.entity.*" 
contentType="text/html;charset=gb18030"%> 
< 
GoodsDAO goodsDao=new GoodsDRO () ; 
Goods goods=goodsDao .findById (request .getParameter ("id")); 
多 > 
<TITLE> 修 改 商 品 信 息 < /TITLE> te tt A 
<FORM action= "dealUpdateGoods .jsp" method="post"> 
<TABLE BORDER=1 ALIGN= "center"> 
区 了 了 R> 


商品 信息 
< /TH> 
</TR> 
<TR ALIGN=CENTER> 
<TD> 
<img src="/images/<%=goods.getImage()%>" height=220 width=180> 
</TD> 
</TR> 
<TR> 
<TD> 
编码 : <%$=goods.getId()%></TD> 
</TR> 
<TR> 
<TD> 
名 称 : 
<input type="text" name="name" id= "name" 
value="<%=goods.getName ()%>"> 
</TD> 


将 待 修改 的 商品 信息 
</TR> 显示 在 相应 的 元 素 


<TR> 
<TD> 
颜色 : 


<input type= "text" name= "color" id="color" 


Value= "<$%$=goods .getColor ()S$> "> 
</TD> 

</TR> 

<TR> 
<TD> 


尺码 : 
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<input type= "text" name= "color" id="color" 
value="<%=goods.getSize()®%>"> 
</TD> 
</TR> 
<TR ALIGN=CENTER> 
<TD> 
价格 : 
<input type= "text" name="price" id="price" 
value="<%=goods.getPrice()®%>"> 


于 
</TD> 
</TR> 
<TR> 


<input type= "hidden" name="id" value=<%=goods.getId()®%>> 
<input type="submit" value=" 修 改 "> 
<input type="reset" value=" 重 置 "> 
</TD> 
</TR> 
</TABLE> 


</FORM> 


执行 效果 如 图 7-3 所 示 。 


= | 年 htpy/ocalhosta080/updateGoodsFormjsp |] [59][x| [下 ,WA ID 


偷 ~ 园 * 咒 " 辐 mp 人 Iaov” 


编码 : JYW12CKB0 

[可 称 :FRR 
车 
尺码 :FL 

| 价格 :Bs.0 “I¥ 
一 


| | |@ intemet | 全 模式 启用 [R100% > 


图 7-3 修改 商品 信息 表单 页 面 


输入 修改 后 的 商品 信息 后 单 击 “ 修 改 ” 按 钮 ,将 把 表单 中 的 参数 交 给 dealUpdateGoods. 
jsp 文件 进行 处 理 , 源 代码 如 下 。 
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闪 示 例 7-4 ”处理 商 品 信息 修改 的 JSP 页 面 dealUpdateGoods. jsp 


<sQ@page language= "java" 
import="java.util.* ,com.jyw.www.service.* ,com.jyw.www.dao.*" 
contentType="text/html; charset=gb18030"%> 
<jsp:useBean id="goods" class="com.jyw.www.entity.Goods" scope="page" /> 
< 各 
request .setCharacterEncoding ("GB18030"); 
response.setContentType ("text/html;charset=GB18030"); 


一 关联 表单 参数 和 实体 bean 属 性 


多 > 
<jsp:setProperty name="goods" property="*" /> 


<jsp:useBean id="goodsDAO" class="com.jyw.www.dao.GoodsDAO" scope="page" /> 


< 一 讽 用 示例 1 编写 的 vpdateQ 方 法， 
goodsDAO.update (goods); 完成 商品 信息 的 修改 


response.sendRedirect ("showGoods.jsp"); 


%> 


dealUpdateGoods. jsp 文件 执行 完毕 后 ,将 跳 回 showGoods. jsp 页 ,读者 可 以 看 到 指 
定 商品 的 信息 已 经 被 修改 了 。 


7.2 任务 二 : 商品 图 片 的 修改 


问题 : 完成 本 章 任务 一 后 ,可 以 修改 商品 的 一 般 信 息 了 ,但 是 如 果 我 们 需要 修改 其 图 
片 , 又 该 如 何 处 理 呢 ? 


在 本 章 任务 一 的 基础 上 ,添加 更 新 商品 图 片 的 功能 。 
”技能 训练 : 
. 。 使 用 java. util. File 类 方法 删除 一 个 文件 。 

。 使 用 button 元 素 传 参数 。 

。 Java 条件 运算 符 的 使 用 。 


7.2.1 添加 修改 商品 图 片 的 入 口 


首先 在 updateGoodsForm. jsp 页 面 中 添加 修改 商品 图 片 的 功能 入 口 。 
关 示 例 7-5 ”添加 更 新 图 片 人 口 后 的 dealUpdateGoods. jsp 


<%@page language="java" 
import="java.util.* ,com.jyw.www.dao.* ,com.jyw.www.entity.*" 
contentType="text/html;charset=gb18030"%> 

< 


GoodsDAO goodsDao=new GoodsDAO (); 
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String id=request.getParameter ("id"); 

Goods goods=goodsDao.findById (id); 

String filename=goods .getImage () 7 

String filechanged=request.getParameter ("filechanged"); 


增加 一 个 新 图 片 文件 名 的 变量 


<TITLE> 修 改 商 品 信息 < /TITLE> 
<FORM name= "forml" action="dealUpdateGoods.jsp" method="post"> 
<TABLE BORDER=1 ALIGN= "center"> 

<TR><TH> 商 品 信 息 </TH>< /TR> 

<TR ALIGN=CENTER> 

<TD> 

<img src="/images/<%=request .getParameter ("filechanged") !=null ? 
request .getParameter ("filechanged") : filenameg>" 
height=220 width=180><br> 


务 > 


<input name= "cbutton" type="button" value=" 更 新 图 片 " 
onClick="document.location.href='updateFile.jsp?id= 
<%=id%>'"><br> 


<font color= "red"><$%=filechanged !=null ?filechanged : ""%><font> 


<input name="image" type="hidden" 
value="<%=filechanged !=null ?filechanged : filenameg> "> 


</TD> 
</TR> 
<TR> 
<TD> 
编码 : <%$=id%>< /TD> 
</TR> 
<TR><TD> 
名 称 : 
<input type="text" name="name" id="name" 
value="<%=goods.getName ()%>"> 
</TD></TR> 
<TR><TD> 
颜色 : 
<input type="text" name="color" id="color" 
value="<%=goods.getColor ()%>"> 
</TD></TR> 
<TR><TD> 


尺码 : 
<input type="text" name="color" id="color" 
Value= "<S$%=goods .getSize()%>"> 
</TD></TR> 
<TR><TD> 
价格 : 
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<input type= "text" name= "price" id="price" 
value="<$=goods.getPrice()®%>"> 

</TD></TR> 

<TR><TD align="center"> 
<input type="hidden" name="id" id="id" value=<%=id%>> 
<input type: 
<input type="reset" value= 

</TD>< /TR> 

</TABLE> 
</FORM> 


初次 进入 updateGoodsForm.jsp 页 时 的 执行 效果 如 图 7-4 所 示 。 


ubmit" value 


重 置 "> 


G© | 功 htpy/ocalhostso8o/updateGoodsFormjsp =] [s+|[X| 于 让 .4350 Pl- 
合 " 国 ”了 坊 ” 队 Pp)v 千 IRO)7” 


图 7-4 添加 了 图 片 更 新 功能 入 口 后 的 修改 商品 信息 表单 页 面 


7.2.2 编写 修改 商品 图 片 相关 的 JSP 文件 


单 击 图 7-4 中 的 “更 新 图 片 ” 按 钮 ,转向 updateFile. jsp。 
闪 示 例 7-6 updateFile. jsp 


<%@page language="java" contentType="text/html;charset=gb18030"%> 

<CENTER> 

<form action="dealUploadFile. jsp" method="post" enctype="multipart/form- 

data"> 

上 传 图 片 <input type="file" name="file"> 
<input type="hidden" name="id" value='<%=request.getParameter ("id")®%>'> 
<input type= “上传 "> 

</form> 
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"submit" value= 


病人 外 htpylocalhosta0so/updatelejspyd=mw- | | a 


上 传 图 片 夺 SESSIONRER [ 桥 汪 


Juploadriejsp |@ Intemet | FP 模式 启用 
图 7-5 上 传 新 图 片 的 JSP 页面 


单 击 图 7-5 中 的 “浏览 …” 按 钮 可 选择 新 图 片 ,选中 图 片 后 , 单 击 
dealUpdateFile. jsp。 


闪 示 例 7-7 dealUpdateFile. jsp 


<%@page language="java" 
import="com.jyw.www.util.* ,com.jspsmart.upload.*" 
contentType="text/html;charset=gb18030"%> 

<% 


“上 传 ”按钮 后 将 转向 


SmartUpload smartupload=new FileManager () .uploadFile (pageContext); 
String filename=smartupload.getFiles() .getFile(0) .getFileName(); 


String id=smartupload.getRequest () .getParameter ("id"); 
%> 
<jsp:forward page="updateGoodsForm.jsp"> 
<jsp:param name="id" value="<$%=id%>" /> 


<jsp:param name="filechanged" value="<%=filename®%>" /> 
</jsp:forward> 


dealUpdateFile. jsp 使 用 JSPSmartUpload 组 件 完成 新 图 片 文件 的 上 传 ,然后 跳 转 回 


商品 信息 修改 页 。 执 行 效果 如 图 7-6 所 示 。 


TO |B npwmoahosaoonpadnekp | [ed] [er , sons [Ea 
丛 * 目 - 山 - 辐 Xp 对 IRolv” 

商品 信息 

更 图 上 
JYW12CKB0. jpg 
编码 : JYW12QKB2 
证 
| 
EE 司 
| 加 Intemet | 于 二 区 100% 了 


图 7-6 上 传 新 图 片 的 JSP 页 面 


135. 


单 击 图 7-6 中 的 “修改 ”按钮 ,仍然 转向 示例 7-4 中 的 dealUpdateGoods. jsp 文件 , 完 
成 商品 信息 的 修改 。 


7.3 任务 三 : 分 店 商品 售 出 后 存货 数量 的 变化 


可 题 : 商品 库存 数量 在 进货 和 售 出 的 时 候 都 应 该 及 时 发 生 增加 或 者 减少 的 变化 ,这 
ee 
”任务 目标 : 
: 商品 数量 应 能 随 着 进货 或 者 售 出 活动 发 生 实时 变化 。 
”技能 训练 : 
: UPDATE 语 语句 的 灵活 使 用 。 


”如果 对 数据 表 中 的 荣 列 的 修改 是 在 原 值 的 基础 上 进行 丧 : 减 ,UPDATE 语句 还 有 一 
适合 这 种 应 用 的 用 法 ,其 语法 格式 如 下 : 


UPDATE 表 名 称 SET 列 名 称 = 新 值 WHERE 列 名 称 = 表 达 式 


7.3.1 编写 StockDAO 类 中 用 于 修改 库存 的 方法 


首先 在 与 库存 信息 数据 表 stock 相对 应 的 We 
的 方法 updateOnOrders() ,该 方法 在 “用 户 订 货 ? 的 情况 下 将 减少 同时 满足 商品 编号 
码 、 货 品 所 在 分 店 为 用 户 提交 的 参数 ,并 且 当 前 库存 量 大 于 订货 量 这 四 个 条 件 的 i 
据 表 中 记录 的 库存 量 ; 在 用户 退货 ”的 情况 下 将 增加 同时 满足 商品 编号 ,尺码 、 货 品 所 在 
分 店 为 用 户 提 交 的 参数 的 记录 的 库存 量 。 源 代码 示例 如 下 。 

闪 示 例 7-8 ”StockDAO 中 的 updateOnOrders() 方 法 


public int updateOnOrders (Orders order) throws Exception { 

InitialContext ctx; 

try { 
ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysqgl"); 
Connection conn=ds.getConnection(); 
PreparedStatement pstmt=null; 
String orderstatus=new String (order.getStatus () .getBytes ("ISO- 
8859-1"),"gb18030"); 
if (orderstatus .equals (" 待 发 货 ") ) 

pstmt=conn.prepareStatement ("UPDATE stock SET stock= stock- ?WHERE 
goodsid=?and storenum=?and size=?and stock>?"); 


如 果 订 单 状 态 改 为 “ 待 发 货 ”， 
则 库存 数量 为 当前 数量 -订货 量 
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else if (orderstatus .equals ("已 退货 ")) 
pstmt=conn.prepareStatement ("UPDATE stock SET stock= stock+?WHERE 


goodsid=?and storenum=?and size=?"); 


如 果 订 单 状态 改 为 “已 退货 ”， 
则 库存 数量 为 当前 数量 + 订货 量 


Pstmt .setInt (1, order.getQuantity()); 
pstmt .setString(2, order.getGoodsid()); 
Pstmt .setInt (3, order.getStorenum()); 
pstmt.setString(4, order.getSize()); 
pstmt.setInt (5, order.getQuantity()); 
return pstmt .executeUpdate (); 

} catch (Exception e) { 
e.printStackTrace (); 

} 


return 0; 


7.3.2 编写 OrdersDAO 类 中 用 于 修改 订单 状态 的 方法 


订单 状态 应 随 着 订货 、 退 货 活动 进行 改变 ,这 就 需要 在 与 订单 信息 数据 表 orders 相 
对 应 的 OrdersDAO 中 添加 修改 订单 状态 的 方法 updateOrderStatus(), 源 代码 示例 如 下 : 
闪 示 例 7-9 OrdersDAO 中 的 updateOrderStatus 〇 方法 


public void updateOrderStatus (Orders order) throws Exception { 

InitialContext ctx; 

try { 
ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds.getConnection(); 
PreparedStatement pstmt=conn.prepareStatement ("UPDATE orders SET 
orderstatus=? ,processdate=?WHERE id=?"); 
String orderstatus=new String (order.getStatus () .getBytes ("ISO- 
8859-1"),"gb18030"); 
Pstmt .setString(1，orderstatus) 二 取 当 前 系统 时 间 | 
java.util.Date ud=new java.util.Date (); 


java.sql.Timestamp stp=new java.sql.Timestamp (ud .getTime ()); 


转变 为 SQL 时 间 惟 类 型 


Pstmt .setTimestamp (2, stp) | 将 数据 表 orders 中 的 订单 
pstmt.setSstring(3, order.getId()); 处 理 时 间 修改 为 当前 系统 时 间 


pstmt .executeUpdate (); 


} catch (Exception e) { 
e.printstackTrace (); 
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7.3.3 编写 OrdersService 类 中 的 update 方法 


订单 的 处 理 过 程 中 会 导致 商品 库存 量 的 变化 和 订单 状态 的 变化 ,涉及 两 个 数据 表 , 所 
以 需要 编写 相应 的 Service 类 的 update 方法 来 进行 处 理 。 
闪 示 例 7-10 ”OrdersDAO 中 的 updateOrderStatus() 方 法 


public Boolean update (Orders order) throws Exception 

{ 
String orderstatus=new String (order .getStatus () .getBytes ("ISO- 8859- 1")，" 
gb18030"); 
if(orderstatus.equals (" 待 发 货 ") ) 


if (new StockDAO () .updateOnOrders (order)>0) 一 -一 在 库存 商品 量 充 足 的 情况 下 ， 
{ 才 去 改变 订单 的 状态 
new OrdersDAO() .updateOrderStatus (order) 
return true; 


; 
else 
return false; 
}else 
new OrdersDAO() .updateOrderStatus (order); 
return true; 


7.3.4 编写 订单 处 理 相 关 的 JSP 文件 


ordersManage. jsp 执行 后 将 列 出 所 有 的 订单 信息 ,并 给 出 处 理 订单 的 功能 入 口 , 源 文 
件 示例 如 下 。 


闪 示 例 7-11 ordersManage. jsp 


<sQe@page language="java" 
import="java.util.* ,com.jyw.www.dao.* ,com.jyw.www.entity.*" 
contentType="text/html;charset=gb18030"%> 
<center> 
订单 列表 
</center> 
<table border=1 align="center" bgColor="#ffffff"> 
<tr> 
<td align=center> 
<B> 序 号 < /B> 
</td> 
<td align=center> 
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<B> 款 号 < /B> 
</td> 
<td align=center> 
<B> 颜 色 < /B> 
</td> 
<td align=center> 
<B> 类 别 < /B> 
</td> 
<td align=center> 
<B> 码 数 < /B> 
</td> 
<td align=center> 
<B> 数 量 < /B> 
</td> 
<td align=center> 


<B> 订 购 日 期 </B> 


</td> 
<td> 


<B> 处 理 日 期 </B> 


</td> 
<td> 

<B> 分 店 < /B> 
</td> 


<td align=center colspan=2> 


<B> 情 况 < /B> 
</td> 


</tr> 


<% 


各 > 


OrderinfoDAO orderinfoDao=new OrderinfoDRO () 
List<Orderinfo>orderinfoList=orderinfoDao.findaAll (); 


for (Orderinfo orderinfo 


: orderinfoList) { 


int quantity=orderinfo.getQuantity(); 


String status=orderinfo.getOrderstatus(); 
Date processdate=orderinfo.getProcessdate(); 
int id=orderinfo.getId(); 


String goodsid=orderinfo.getGoodsid(); 


String size=orderinfo.getSize(); 


<tr> 
<td align=center><%=id%></td> 
<td align=center><%=goodsid%></td> 


<td align=center><%=orderinfo.getColor()%></td> 
<td align=center><%=orderinfo.getName ()%></td> 


<td align=center><%=size%></td> 
<td align=center><%=quantity%></td> 
<td align=center><%=orderinfo.getApplydate ()%></td> 


<td align=center><%=processdate !=null ?processdate : ""%></td> 


扩 的 埠 式 各 出险 | 
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<td align=center><%=orderinfo.getstorenum()®%></td> 
< 
if (status.equals ("已 发 货 ") 11 status.equals ("已 取消 ")) { 
多 > 
<td colspan=2 align=center> 
<$%=status®> 
</td> 
<% 
} else if (status .equals(" 待 发 货 ") ) { 
多 > 
<td align=center> 
<input type="button" value=" 发 货 " 
onClick="document. location.href= 'ordersConfirm. jsp? id=<%= 
id%>&status= 已 发 货 '"> 
</td> 
<% 
}elsef{ 
$> 
<td> 
<input type="button" value= "确认 " 
onClick="document .location.href='ordersConfirm.jsp?id=<%= 
id%>&goodsid=<%=goodsid%> &size=<%=size%>&storenum= 
0gquantity=<%$=quantity%> &status= 待 发 货 '"> 


utton" value= "取消 " 
onClick="document .location.href='ordersConfirm.jsp?id=<%= 
ids>&status= 已 取消 '"> 


</td> 
< 和 


多 > 
<td align=center> 
<formaction="orderquery.jsp" name="form1l"> 
<input type="hidden" name="stock" 
value="<%=orderinfo.getQuantity()$>"> 
<input type="hidden" name= "orderid" 
value="<%=orderinfo.getId()®>"> 
<input type="hidden" name= "goodsid" 
value="<%$=orderinfo.getGoodsid()®%>"> 


hidden" name="situid" value 
hidden" name= 


<input type= 


<input type= handle" value= 
</form> 
</td> 


< 和 


$%> 
</tr> 
</table> 


执行 后 的 效果 如 图 7-7 所 示 。 
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Ge ~ |@ hitpi/ocalhoste080/ordersManagejsp EE | 四 | 日 


帘 突 外 hupylocahostaoaorordesManagejsp | | 同 ~ 轩 -各 -四 mmEnv 夺 IEROv” 
习 


订单 列表 
序号 | 款 号 ”| 颜色 | 类 别 码 数 | 数 量 订购 日 期 处 理 日 期 ”分店 | 情况 
S 
X 
M 
S 


JYW12CKN1 | 水 洗 黑 | 千 仔 祥 10 2013-02-222013-02-22| 1 EJ| 
JYW12CKN1 水 洗 黑 | 牛仔 祥 20 2013-02-22 | 
[本 | 3 潭 


TYW12CKN1 水洗 黑 | 年 仔 祥 2 2013-03-31 1 
JYW12CKB0| 蓝 | 牛仔 祥 2 2013-03-31 国医 


mon- 


7-7 订单 列表 页 


单 击 图 7-7 中 的 功能 按钮 ,将 转向 ordersConfirm. jsp 文件 ,其 源 文件 示例 如 下 。 


闪 示 例 7-12 ordersConfirm. jsp 


<%Q@page language="java" import="java.util.* ,com.jyw.www.service.*" 
contentType="text/html;charset=gb18030"%> 
<jsp:useBean id="orders" class="com.jyw.www.entity.Orders" scope="page" /> 


<jsp:setProperty name= "orders" property="*" /> 
<% 
if (!new OrdersService () .update (orders)) { 
out .Print ("订单 确认 失败 ,请 检查 库存 量 是 否 足够 "); 
out .print ("<input type= 'button' value=' 确 定 ' 
onclick=document .location.href='ordersManage.jsp'>"); 


} else 
response.sendRedirect ("ordersManage.jsp"); 


和 > 
单 击 图 7-7 中 的 某 笔 订单 操作 栏 的 “确定 ”按钮 ,将 转向 ordersConfirm. jsp 文件 ,如 
果 当 库存 量 大 于 订货 数量 时 ,执行 后 自动 跳 转 回 如 ordersManage. jsp 页 ,相应 订单 的 操 
作 栏 改变 为 "发 货 ? 按 钮 ,如 图 7-8 所 示 。 

| 合同 -epyoenoxaooordowoooim Blssr.sw 上 


窗 从 看 htpi/ocalhostscs0/checklogin 丛 ~ 加 -由 v 轧 FEiplv 伟 IROlv” 


订单 列表 
款 号 码 数 数量 | 订购 日 期 处 理 日 期 操作 


TYWI2CKN1 | 10 2013-02-22|2013-02-22 Ea 
JYW12CKN1 M | 20 |2013-02-22|2013-04-05 
JYW12CKN1 | 2 |2013-03-31|2013-04-05 已 取消 | 
JYW12CKBO i | 2 |2013-03-31 育 训 | 双 注 | 司 
[TT[TTT Orilemeas [3 


图 7-8 库存 量 不 能 满足 订单 要 求 时 的 提示 


如 果 库 存量 小 于 订货 数量 时 ,OrdersService(). update() 方 法 返回 false, 这 种 情况 下 ， 
ordersConfirm. jsp 将 显示 相应 的 提示 信息 ,如 图 7-9 所 示 , 单 击 图 7-8 中 的 “确定 ”按钮 将 
返回 ordersManage. jsp 页 ,操作 栏 保持 不 变 。 

单 击 图 7-7 中 的 某 笔 订单 操作 栏 的 “取消 ?或 者 “发 货 ? 按 钮 ,将 转向 ordersConfirm. 
jsp 文件 ,执行 后 自动 跳 转 回 如 ordersManage. jsp 页 ,相应 订单 的 状态 改 为 “已 取消 "或 者 
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“已 发 货 ”, 如 图 7-10 所 示 。 


SO- [Broranosramoordersconmyspne=20g00dsd= mwizcenu | [S|] 下 -Oo EH 
宣 安 急 http//ocalhostao8o/ordersconfimjsp?d=.. | | 住 >~ 且 7 了 坟 ” 忆 iP)v 合 IRO)v” 
人 习 
发 货 失败 ， 请 检查 库存 量 是 否 足够 邑 寺 | 
贺 
3 IT 开本 Rl00% 了 


图 7-9 库存 量 不 能 满足 订单 要 求 时 的 提示 


SO |B hepwoohoreoeoordersvonoejm ER 
窗帘 加 htpylocahos:acaolcheckdogm jml 全- 加- 品 - 电 rmp -人 TIRov” 
可 


订单 列表 
款 号 | 颜色 | 类别 码 数 数量 | 订购 日 期 | 处 理 日 期 分店 | 操作 
JYW12CKN1 | 水洗 黑 | 牛 仔裤 | S | 10 |2013-02-22|2013-02-22| 1 | | 
[JYW12CKN1 | 水 洗 黑 年 仔 裤 | M | 20 [2013-02-22|2013-04-05| 1 | 已 发 货 | 
[JYW12CKN1 | 水 洗 黑 咎 仔裤 | M | 2 |2013-03-31|2013-04-05| 1 | 已 取消 | 
TYW12CKB0| 蓝 ”年 仔裤 | S | 2 |2013-03-31 | 1 ym 悦 
[| [ [TT Brenel er en EE 


图 7-10 库存 量 不 能 满足 订单 要 求 时 的 提示 


7.4 任务 四 : 商品 售 亏 后 信息 的 删除 


问题 : Web 项 目 运行 一 段 时 期 后 ,会 积累 相当 规模 的 数据 ,对 于 不 再 有 保存 价值 的 数 
据 ,应 当 及 时 删除 ,以 减少 存储 空间 的 开销 。 例 如 : 随 着 商品 的 售 出 ,商品 库存 数量 在 不 
断 减少 , 当 商 品 售 殴 并 停产 后 ,这 条 商品 的 信息 就 可 以 被 删除 了 。 那 么 项 目 中 需要 对 数据 
表 中 记录 进行 删除 的 功能 该 如 何 实现 呢 ? 


“任务 目标 
| 将 用 户 指 定 的 商品 的 信息 从 goods 数据 表 中 删除 。 


技能 训练 ， 


7.4.1 GoodsDAO 中 添加 deleteByld 的 方法 


GoodsDAO. java 中 deleteById() 方 法 用 于 删除 指定 编号 商品 的 分 店 售 亏 商品 的 信息 。 
闪 示 例 7-13 ”GoodsDAO. java 中 的 deleteById() 方 法 


public void deleteById(String id) throws Exception { 
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第 7 章 商品 信息 的 修改 和 删除 


InitialContext ctx; 
LL 
ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysq1"); 
Connection conn=ds.getConnection(); 
PreparedStatement pstmt=conn 
.PrepareStatement ("DELETE FROM goods WHERE id=?"); 
pstmt .setstring(1, id); 
pstmt .executeUpdate (); 
} catch (Exception e) { 
e.printstackTrace (); 


7.4.2 编写 删除 商品 信息 相关 的 JSP 页 


在 示例 5-5 显示 商品 信息 页 showGoods. jsp 中 添加 “删除 ?功能 人口 ,代码 如 下 : 


<input type="button" value=" 删 除 " 
onClick="document .location.href='delGoods.jsp?id=<%=goods.getId()®%>'"> 


修改 后 ,显示 商品 信息 页 执行 效果 如 图 7-11 所 示 。 


GO- http://locahocta080/showGoodsjsp 7] Ea mF. cnn 


窗 六 ”由 httpy/localhosr8080/showGoodsjsp 了 师 ~ 四 FEPlv 全 IRov” 


NA 


JYW12QKB2 JYW12CKN1 JYW12CKY2 
牛仔裤 2 牛仔 祥 | 牛仔 裤 

蓝 5-L [¥98.0 “| 蓝 |S-XL[¥100.0 || 水 洗 黑 [S-XL ¥123.0|| 黄 |S-XL 兰 158.0 
修改 ] 出 除 ] 修改 | 则 只 | 褒 次 |】 出 除 ] 从 小] WW3 | 


E&I 


图 7-11 添加 了 “删除 ”功能 入 口 后 的 商品 信息 显示 页 


单 击 图 7-11 中 的 “删除 ”按钮 ,将 转向 delGoods. jsp, 甚 源 文件 示例 如 下 : 
闪 示 例 7-14 delGoods. jsp 


<%@page language="java" import="java.util.* ,com.jyw.www.service. * ,com.jyw. 
Www.dao.*™" 

contentType= "text/html; charset=gb18030"%> 

< 多 

new GoodsDRO () .deleteById (request .getParameter ("id")); 

response.sendRedirect ("showGoods .jsp") 7 

$> 
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7.5 知识 扩展 


7.5.1 修改 所 有 行 


SQL UPDATE 语句 可 以 不 写 条 件 部 分 ,格式 如 下 : 
UPDATE table name SET 列 1= 值 1,..., 列 n= 值 n 


这 种 情况 下 ,数据 表 中 所 有 记录 的 指定 列 都 将 被 修改 为 指定 的 值 。 


7.5.2 删除 所 有 行 


SQL DELETE 语句 也 可 以 不 写 条 件 部 分 ,格式 如 下 : 
DELETE FROM table name 

或 者 
DELETE * FROM table name 


这 种 情况 下 ,将 在 不 删除 表 的 情况 下 删除 所 有 的 记录 。 表 的 结构 .属性 和 索引 仍 保持 
完整 。 


课 后 练 习 


1. 本 章 任务 二 中 完成 了 商品 图 片 的 更 新 ,但 是 商品 的 原 图 片 仍然 在 /Image 目录 下 ， 
浪费 存储 空间 ,请 在 任务 二 基础 上 ,补充 删除 原 图 片 文件 的 功能 。 

提示 : 使 用 java. util. File 类 的 相应 方法 。 

2. 为 “ 佳 衣 屋 "项目 添加 “退货 "功能 。 

要 求 : 分 店 可 向 总 店 申请 “退货 ”, 并 提交 退货 商品 的 编号 .尺寸 .退货 量 等 信息 ;总 店 
在 订单 显示 页 中 可 看 到 分 店 申请 的 “退货 " 单 ,并 可 完成 退货 操作 。 

提示 : 参考 本 章 任 务 三 完成 。 

3. 为 “ 佳 衣 屋 ”项 目 添加 “删除 ”员工 信息 的 功能 。 

要 求 : 删除 指定 的 员工 信息 。 

提示 : 参考 本 章 任 务 四 完成 

4. 本 章 学 习 的 主要 内 容 为 : 修改 和 删除 数据 表 中 的 记录 。 请 结合 第 1 章 课 后 练习 的 
要 求 , 完 成 项 目 中 相应 的 修改 或 者 删除 数据 表 中 记录 的 功能 模块 。 
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系统 的 安全 设计 


第 8 章 账号 安全 控制 


本 章 学 习 要 点 : 

。 熟练 掌握 servlet 的 生命 周期 ; 

。 熟练 掌握 servlet 的 配置 ; 

。 熟练 用 servlet 处 理 cookie; 

。 熟练 掌握 servlet 会 话 追 踪 的 功能 ; 

。 熟练 掌握 Filter 过 滤器 的 编写 和 配置 方法 ; 

。 熟练 掌握 Listener 监听 器 的 编写 和 配置 方法 。 


完成 前 7 章 的 学 习 后 ,项 目 已 经 实现 了 许多 功能 模块 ,这 些 功 能 均 涉 及 数据 表 增 、 删 、 
改 、 查 。 但 是 这 些 功 能 都 是 对 所 有 人 开放 的 ,这 显然 不 符合 系统 的 安全 性 要 求 。 在 第 2 章 
需求 分 析 中 已 经 明确 : 系统 用 户 有 三 种 角色 , 即 管理 员 ,总 店 店员 和 分 店 店员 ,三 种 角色 
应 有 不 同 的 操作 权限 ,本章 将 完善 系统 的 账号 安全 控制 。 


8.1 任务 一 : 用 户 登 录 功 能 的 实现 


; 任务 目标 : 
用 户 输入 用 户 名 、 密 码 ,选择 权限 ,如 果 上 述 信 息 都 正确 , 则 跳 转 到 各 角色 下 的 默 
; 认 主页 ,否则 提示 登录 错误 信息 。 | 
”技能 训练 : 
| 。 Sevlet 的 编写 。 

。 Sevlet 的 配置 。 


在 本 章 中 引入 一 种 新 的 技术 一 一 Servlet 技术 来 完成 这 一 任务 。 


8.1.1 Servlet 简介 


要 实现 用 户 登 录 功 能 ,首先 要 了 解 一 种 特殊 的 Java 类 一 一 Servlet。 与 其 他 的 Java 
相 比 , 它 的 特殊 点 在 于 ,Servlet 必 须 继承 Servlet 类 或 者 是 它 的 子 类 .通常 是 HttpServlet 


中 小 型 Web 项 目 开发 实战 


个 类 。 在 Web 应 用 中 , Servlet 接受 来 自 浏览 器 的 请 求 , 读 取 从 客户 端 发 送 过 来 的 数 
ee ye 
户 端 。 在 下 面 的 小 节 中 采用 JSP+ Servlet 完成 用 户 登 录 功 能 。 


8.1.2 MyEclipse 自动 生成 Servlet 


首先 借助 MyEclipes 的 向 导 功 能 完成 一 个 Servlet 的 创建 ,如 图 8-1 一 图 8-5 所 示 。 
(1) 新 建 一 个 程序 包 
(1) 右 击 项 目 名 ”(2) 单 击 菜单 项 的 New 命 令 


9 l 
[ER 区 TB Project 


各 
外 6 Into 鲜 Enterprise Application Froject 
外 Open in ker Vindow 前 Yeb Project 
国 ，。 ”0pea Type Wierarchy Fa 六 eb Service Project 
和 Show In LttShiftt ?大 Java Project 

| Report Web 了 了 

力 cery CerltC 办 Mport Web Pojee 
四 到 Project. 
| 自贡 国 copz Qualified Nane 
由 -可 让 上 :te Ctrlty 


(3) 单 击 级 联 菜 单 中 的 Package 
图 8-1 新 建 包 


9 Hew Java Package 


Java Package 
Create a new Java package, | 


Creates folders corresponding to packages. 


Source folder: |jyw/src Browse. 


Nane: com jy www. servlet 一 


(D 输入 包 名 ， 建 议 最 后 
一 级 包 名 为 servlet 


四 0 


@ 
图 8-2 定义 包 名 
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(2) 新 于 


BE Servlet 


(1) 右 击 包 名 


(2) 单 击 菜单 项 


站 汪 安 全 把 制 | 


hh 的 New 命 令 


om 
册 

册 co jyw. www oti]| 
Bi JEE System Library 
Bi Java EE 5 Librarie: 

Bi Referenced Taber 
窜 YebRoot 
jy sa 


忆 imz 


四 


Inage Previe 2 、 9 sm 


| 总 Enterprise Application Project 
戎 Yeb Project 


Open in Ben Window 


Dpen Type Hierarchy 了 4 


着 Yeb Service Project 
蓝 Java Project 


Show In 各 ttShifttW | 国 Report Web Project 
Guiic | [Project... 
区 copz Qualified Hane 密 Packsee 
合 Paste Ctrlty © rss 
其 Delete Delete @ Interface 
Build Path rcsTolder: 
Souree 如 ttShifttS ?| 贿 芭 攻 
Refactor 如 ttShifttT *| DFile 
一 Pe | 区 Applet 
Ba Import 四 mL Mdvanced Tenplates) 
LA Expsrt 


只 Refresh 
Assign Working Sets. 
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| 古 JSP CAavanced Tenplates) 


(1) 输入 Servlet 的 名 字 


(2) 勾 选 需要 的 方法 ， 
一 般 保持 默认 
选项 即 可 


(3) 单 击 Next 按 钮 


(3) 
图 8-3 ”新建 servlet 


Create a new Servlet. 


Servlet Wizard 


Craate a new Servlet class. 


单 击 级 联 菜 单 9 


hb 的 Servlet 命 邻 


Source folder- jywysre 


Browse 


om _ jyw_www_serwlet 


Browse. 


| Broxse, 


Checkl ogin 


Oblie Odfayl 
Dabstract Dfing 


4 Oprizate 人 protecteda 


Deooe 


javax. servlet http. HttpServlet 


图 8-4 


新 建 Servlet 属性 的 设置 
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Create 3a me 可 ServJlet- 


Generate/Map web. xml file 


ServlatJSP class None: 


Servlet/JSP Name; ICheckl ogin 


Servlet/JSP Mapping URL: /servlet/Checkl ogin 


File Path of web. xml: fjyw/WebRoot/WEB-INF Browse... 


Display Nane: This is the display nane of my 


Description; This is the description of my J| 


单 击 Finish 按 钮 


图 8-5 完成 新 建 servlet 的 操作 
完成 向 导 引 导 的 所 有 步骤 后 , MyEclipse 会 自动 生成 如 下 的 CheckLogin. java 文件 。 
闪 示 例 8-1 CheckLogin. java 源 代码 


package com.jyw.www.servlet; 


import java.io.IOException; 
import java.io.PrintWriter; 


import javax.servlet.ServletException; 

import javax.servlet.http.HttpServlet; 

import javax.servlet.http.HttpServletRequest; 
import javax.servlet.http.HttpServletResponse; 


public class Checklogin extends HttpServlet { 


Servlet 必 须 继承 Servlet 类 或 者 是 
J 它 的 子 类 ， 通 常 是 HttpServlet 这 个 类 
* Destruction of the servlet. <br> 


*/ 
public void destroy() { 二 自动 生成 的 destroy 方 法 


super.destroy(); // Just puts "destroy" string in log 
// Put your code here 


/x 
* The doGet method of the servlet. <br> 


关 


x* This method is called when a form has its tag value method equals to get. 
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Qparam request the request send by the client to the server 
@param response the response send by the server to the client 


第 可 党 本 


x Q@throws ServletException if an error occurred 
*x @throws IOException if an error occurred 
»/ 
public void doGet (HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException { 


response.setContentType ("text/html"); 

PrintWriter out=response.getWriter (); 

out.println("< !DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 
Transitional//EN\">"); 

out.println ("<HTML>"); 

out.println(" <HEAD><TITLE>A Servlet</TITLE></HEAD>"); 

out.println(" <BODY>"); 

out .Print (" Th Li 

out.print (this.getClass ()); 

out.println(", using the GET method"); 

out .Println("” </BODY>" 

out .Println("</HTML>") 7， 

out .flush (); 

out.close(); 


* The doPost method of the servlet. <br> 
* This method is called when a form has its tag value method equals to post. 


# @param request the request send by the client to the server 
* @param response the response send by the server to the client 
* @throws ServletException if an error occurred 


x* @throws IOException if an error occurred 
x/ 3 自动 生成 的 doPost 方 法 


public void doPost (HttpServletRequest request,HttpServletResponse response) 
throws ServletException, IOException { 


response.setContentType ("text/html"); 

PrintWriter out=response.getWriter(); 

out.println ("< !DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.01 
Transitional//EN\">"); 

out.println ("<HTML>"); 

out .Println(" <HEAD><TITLE>A Servlet</TITLE></HEAD>"); 

out.println(" <BODY>"); 

out.print(" Th 

out .print (this.getClass()); 

out .Println("，using the POST method"); 

out .println(" </BODY>"); 

out .Println("</HTML>") 7 

out .flush() 


out .close (): 
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/ 关 关 
* Initialization of the servlet. <br> 
其 
* @throws ServletException if an error occurs 
*/ 
public void init () throws ServletException { 
// Put your code here 


自动 生成 的 init 方 法 


} 


MyElipse 自动 生成 的 代码 重 写 了 HttpServlet 中 的 doPost() 方 法 ,使 得 它 可 以 处 理 
客户 端的 Post 方式 的 请 求 。doPost() 方 法 有 两 个 参数 ,分 别 为 HttpServletRequest 和 
HttpServletResponse, 这 两 个 参数 分 别 用 于 表示 客户 端的 请 求 和 服务 器 端的 响应 。 通 过 
HttpServletRequest, 可 以 从 客户 端 中 获得 很 多 客户 端 发 送 过 来 的 信息 , 通过 
HttpServletResponse 服务 器 可 以 向 客户 端 发 送信 息 。 


8.1.3 编写 EmployeeDAO 类 中 的 isEmployee 方法 


要 完成 本 章 任 务 一 ,需要 验证 客户 端 提 交 的 账号 .密码 及 权限 是 否 与 数据 表 
employee 中 的 记录 一 致 ,下 面 在 EmployeeDAO 中 添加 一 个 方法 来 完成 判断 ,这 种 方法 
返回 值 为 布尔 型 ,方法 命名 规范 一 般 以 is 开头 ,格式 如 下 : 


public Boolean isXxx (Xxx xxx) 


按照 命名 规范 为 EmployeeDAO 添加 一 个 isEmployee() 方 法 , 源 代码 示例 如 下 。 
闪 示 例 8-2 源 代 码 EmployeeDAO 类 中 的 isEmployee() 方 法 


public Boolean isEmployee (Employee employee) 
throws Exception { 
InitialContext ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds.getConnection(); 
PreparedStatement pstmt=conn.prepareStatement ( 
"SELECT * FROM employee WHERE account=?and 


password=?and authority=?", 查找 数据 表 employee 中 
ResultSet .TYPE SCROLL INSENSITIVE, 账号 、 密 码 和 权限 都 与 
ResultSet .CONCUR READ ONLY); 用 户 提交 的 值 相同 的 记录 


pstmt.setString (1, employee.getAccount ()); 
pstmt .setString(2，employee.getPassword () ) 
Pstmt .setInt (3, employee.getAuthority()); 
ResultSet rs=pstmt .executeQuery (); 


if (rs.first()) { 
return true; 一 一 如 果 有 满足 条 件 的 记录 ， 则 返回 true 
} else { 
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return false; 


8.1.4 改写 doPost 和 doGet 方法 


对 自动 生成 的 doPost( ) 方 法 进行 改写 ,首先 要 完成 用 户 提交 信息 的 接收 ,然后 调用 
8. 1.3 小 节 中 编写 的 isEmployee() 方 法 判断 用 户 提 交 的 信息 是 否 正确 ,再 根据 情况 作 相 
应 跳 转 ,并 使 用 session 保存 用 户 已 登录 的 状态 ,如 示例 8-3 所 示 。 

闪 示 例 8-3 ”改写 后 的 doPost() 方 法 


public void doPost (HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException { 
request .setCharacterEncoding ("GB18030"); 
response.setContentType ("text/html;charset=GB18030"); 
EmployeeDAO employeeDao=new EmployeeDAO () 


Employee employee=new Employee(); 
String account=request .getParameter ("account"); 


employee.setAccount (account); 
String password=request .getParameter ("password"); 


employee.setPassword (password); 
Integer authority= Integer.parseInt (request .getParameter ("authority ")); 


employee.setAuthority (authority); 


if (employeeDao.isEmployee (employee)) { 
. oy) (人 恨 所 用 户 权限 中转 到 不 同 的 页 而 


Switch (authority) { 


case 0: 


case 1: 


case 2: 


} 


} else { 


request .getRequestDispatcher ("/showSales.jsp") .forward (request, 


response); 
request .getSession() .setRAttribute ("adminAuth", account); 


break; // 管 理 员 


| 使 用 session 保 存 用 户 已 登录 的 状态 


request .getRequestDispatcher ("/showOrders .jsp") .forward (request, 
response); 

request .getSession().setAttribute("account", account); 

break; // 总 店 店员 

request .getRequestDispatcher ("/searchStock.jsp") .forward 
(request, response); 

request .getSession () .setAttribute ("account", account); // 分 店 店员 


request .getRequestDispatcher ("/error.jsp") .forward (request, response); 


} 


为 了 避免 用 户 直接 跳 过 登录 页 面 而 在 浏览 器 中 输入 地 址 加 参数 的 形式 来 登录 ,可 以 
将 doGet() 改 写 为 示例 8-4 所 示 的 处 理 方式 。 
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关 示 例 8-4 改写 后 的 doGet() 方 法 


Public void doGet (HttpServletRequest request,HttpServletResponse response) 
throws ServletException,IOException 


PrintWriter out=response.getWriter (); 


out.println(" 请 从 登录 页 面 登录 "); 


8.1.5 ”配置 servlet 映射 


Servlet 编写 完成 后 ,为 了 能 够 从 JSP 中 使 用 Servlet, 还 必须 在 项 目的 Web-INF 目录 下 
的 Web. xml 中 对 它 进行 配置 。 对 Servlet LoginCheck 的 配置 ,是 在 Web. xml 的 二 Webapp 过 
< 过 /Webapp 过 标签 之 间 插 人 示例 8-5 所 示 的 子 标签 。 

闪 示 例 8-5 servlet 的 配置 


<servlet> 


servlet 别 名 不 能 含 空格 。 
<servlet-name>LoginServlet</servlet- name> 一 一 配置 别名 ， 可 以 避免 将 servlet 的 
实际 路 径 暴 露 给 客户 端的 用 户 

<servlet-class>com.jyw.www.servlet.LoginCheck </servlet-class> 
</servlet> 


<servlet-mapping> 指明 有 此 别名 的 Servlet 类 


<servlet-name>LoginServlet </servlet-name> 
中 访 让 为 路 外 
0 bern /en /i enn JSP 中 访问 此 servlet 的 路 径 


</servlet-mapping> 


8.1.6 编写 登录 页 login.jsp 


Servlet 编写 配置 完成 后 ,就 可 以 在 JSP 文件 中 访问 它 了 。 示 例 8-6 是 “ 佳 衣 屋 " 系 统 
的 登录 页 面 的 源 代码 ,注意 表单 的 action 属性 使 用 了 servlet 访问 路 径 。 
闪 示 例 8-6” 源 代码 login. jsp 


<%@page language="java" import="java.util.*" 
contentType="text/html;charset=gb18030" pageEncoding= "gb18030"%> 
<center> 


<form name="forml" method="post" action="/checklogin"> 


用 户 名 Web.xml 中 配置 的 
a type="text" name="account" size="15"> i 
密码 

<input type="password" name="password" size="15"> 

<br> 

选择 身份 


<select name="authority"> 


<option value="0"> 
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管理 员 
</option> 
<option value="1"> 
总 店 店员 
</option> 
<option value="2"> 
分 店 店员 
</option> 
</select> 
<br> 
<input type="submit" value=" 登 录 " /> 
</form> 
</center> 


登录 页 的 执行 效果 如 图 8-6 所 示 。 


-jsix| 
相国 -|@@htpyocahoraoeooonip 可 问 罗 [了 一, [g 


帘 倪 ” 世 htpV/localhostsoso/login- 丛 ~ 园 ~ 权 ~ 思 REp vv 对 IRov” 
用 户 名 


密 码 
选择 身份 [和 宣 员 悦 
如 | 


[Ti [TT fmemetl Sm eg Peis 


图 8-6 登录 页 的 执行 效果 


8.2 任务 二 : 验证 码 的 生成 


问题 : 我 们 在 登录 一 些 Web 应 用 ,如 网 银 、 电 子 邮 件 等 系统 时 ,系统 除了 要 求 输入 用 


户 名 和 密码 外 ,还 往往 要 求 输入 一 个 “验证 码 ”, 这 是 为 什么 ?又 如 何在 我 们 的 系统 中 实现 
这 一 功能 呢 ? 


许多 黑客 工具 使 用 穷 举 法 来 破解 用 户 的 密码 ,输入 验证 码 是 为 了 防止 有 恶意 的 用 户 


使 用 自动 注册 机 等 工具 来 不 停 地 进行 登录 尝试 。 验 证 码 ,就 是 在 每 次 访问 登录 界面 时 , 随 
机 生成 一 串 新 的 字母 ,数字 甚至 中 文 ,一 方面 将 这 个 验证 码 生成 图 片 显示 给 用 户 ; 另 一 方 
面 将 它 保存 在 服务 器 上 , 当 用 户 进行 登录 的 时 候 , 输 入 验证 码 , 通 过 比较 用 户 输入 的 验证 
码 和 保存 在 服务 器 上 的 验证 码 是 否 一 致 ,判断 用 户 是 否 在 恶意 使 用 登录 功能 。 使 用 图 片 
文字 显示 验证 码 主要 是 为 了 防止 有 些 工 具 能 够 通过 分 析 网 页 而 得 到 验证 码 。 


8.2.1 编写 生成 验证 码 图 片 的 servlet 


本 小 节 中 使 用 servlet 来 产生 随机 的 四 位 数 验 证 码 , 并 且 将 这 四 位 验证 码 以 图 像 的 形 
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式 输出 到 客户 端 。 
闪 示 例 8-7 产生 图 像 验 证 码 的 CreatValidationCode. java 


package com.jyw.www.servlet; 


import java.io.*; 

import javax.servlet.*; 
import javax.servlet.http.*; 
import java.util.Random; 
import java.awt.*; 

import java.awt.image.*; 
import javax.imageio.*; 


public class CreatValidationCode extends HttpServlet 
{ 
Public void doPost (HttpServletRequest request, 
HttpServletResponse response) 
throws ServletException,IOException 
' 
response.setContentType ("image/jpeg") 
response.setHeader ("Pragma", "No-cache 


response.setHeader ("Cache-Control", "no-cache"); Re 


response.setDateHeader ("Expires",0); 使 用 缓存 里 的 图 像 


OutputStream out=response.getOutputStream(); 
int width=80, height=20; 
BufferedImage image=new Dateroo nag (ws height, 
BufferedImage.TYPE INT RGB); 了 
= 设置 缓冲 图 片 的 宽 和 高 


Graphics g= image .getGraphics (); 
得 到 Graphics 对 象 ， 以 绘制 


Random random=new Random () 生成 随机 类 


g.fillRect (0, 0, width, height); 
py Ci S| 设置 字体 


g.setFont (new Font ("Times New Roman", Font .ITALIC, 18)); 


String sRand: RE 
for (int i=0;i<4;i++){ 产生 一 个 随机 数 


String rand=String.valueOf (random.nextInt (10)); 
sRand+=rand; / 设置 随机 颜色 


国 


像 


g.setColor (new Color (20+random.nextInt (110),20 
+random.nextInt (110),20+random.nextInt (110))); 


g.drawString (rand, 20* i+6,16) i 绘制 图 片 


g.dispose(); 


// 
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ImageIO.write (image, "JPEG", out); 
将 缓冲 
public void doGet (HttpServletRequest request, 
HttpServletResponse response) 
throws ServletException,IOException 


输出 到 页 面 


doPost (request, response); 


) “pos 方式 相同 


8. 


DD 


.2 验证 码 图 片 servlet 的 配置 


在 Web. xml 的 二 Web-apps 二 中 插入 下 列 代码 。 


<servlet> 
<servlet-name>CreatValidationCode</servlet-name> 
<servlet-class>com.jyw.www.servlet.CreatValidationCode< /servlet-class> 
</servlet> 
<servlet-mapping> 
<servlet-name>CreatValidationCode</servlet-name> 
<url-pattern> /creatValidationCode< /url-pattern> 
</servlet-mapping> 


8.2.3 使 用 验证 码 图 片 生成 的 servlet 


下 面 改写 示例 8-6, 添 加 图 片 验证 码 到 登录 表单 中 。 
闪 示 例 8-8 添加 了 图 片 验证 码 后 的 login. jsp 


<%epage Language= "java" import="java.util. 关 " 
contentType="text/html;charset=gb18030" pageEncoding= "gb18030"%> 
<center> 
<form name="form]l" method="post" action="/checklogin"> 
用 户 名 
<input type="text" name="account" size="15"> 
<br> 
密码 
<input type="password" name="password" size="15"> 
<br> 
验证 码 
<input TYPE= "text" size="10" name= "valCode"> 
<img src="/creatValidationCode"> 
<br> 


选择 身份 使 用 验证 码 生成 servlet 


<select name="authority"> 


<option value="0"> 
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管理 员 
</option> 
<option value="1"> 
总 店 店员 
</option> 
<option value="2"> 
分 店 店员 
</option> 
</select> 
<br> 
<input type="submit" value=" 登 录 " /> 
</form> 
</center> 


在 这 个 HTML 程序 中 ,定义 了 一 个 表单 ,用 于 接收 用 户 的 登录 名 和 密码 ,另外 ,在 这 
两 个 表单 元 素 的 基础 上 ,加 上 了 一 个 验证 码 的 输入 框 ,用 户 必 须 在 登录 的 时 候 输 入 这 个 验 
证 码 , 如 图 8-7 所 示 。 


验证 码 厂 一 一 3080 
选择 身份 [BR 可 
Ey 全 


| |@ Intemnet | 晤 模式 局 用 | 下 100% > 


图 8-7 带 验证 码 的 登录 页 


8.2.4 校对 验证 码 


下 面 进一步 修改 示例 8-3, 加 入 对 验证 码 的 校对 。 
闪 示 例 8-9 添加 验证 码 校对 功能 的 doPost( ) 方 法 


public void doPost (HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException { 
request.setCharacterEncoding ("GB18030"); 
response.setContentType ("text/html;charset=GB18030"); 
EmployeeDAO employeeDao=new EmployeeDAO (); 
Employee employee=new Employee (); 
String account=request .getParameter ("account"); 
employee.setAccount (account); 
String password=request .getParameter ("password"); 
employee.setPassword (password); 
Integer previlege= Integer.parseInt (request .getParameter ("authority")); 
employee.setAuthority (previlege); 
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String code= (String) sess.getAttribute ("code"); 获取 生成 的 验证 码 
String valCode=request .getParameter ("valCode"); 


ty 获取 用 户 输入 的 验证 码 
SE: (valCode na valCode .equals (code)) 1 一 一 验证 码 的 校对 | 
if (employeeDao.isEmployee (employee)) { 
switch (previlege) { 
case 0: request .getRequestDispatcher("/showSales.jsp") .forward 
(request, response); 
break; // 管理 员 
case 1: request .getRequestDispatcher("/showOrders.jsp") .forward 
(request, response); 
break; // 总 店 店员 
case 2: request .getRequestDispatcher("/searchStock.jsp") .forward 
(request，response); // 分 店 店员 


}else { 
request .getRequestDispatcher ("/error.jsp") .forward (request,response)7 
} 
lelse{ 
PrintWriter out=response.getWriter (); 
out .print ("验证 码 错误 "); 
out.print ("<a href='/login.jsp'>"+" 回 登录 页 "+"</a>"); 
out.flush(); 
out.close(); 
} 
} catch (Exception e) { 
e.printStackTrace () 7 
} 
} 


修改 完成 后 ,就 可 以 顺利 实现 验证 码 功能 了 。 


8.3 任务 三 : 用 户 名 和 密码 在 客户 端的 保存 


问题 : 打开 “淘宝 ”登录 页 ,在 输入 密码 的 右 下 方 有 一 个 “十 天 免 登 录 ” 选 项 ,如 果 勾 选 
了 这 个 选项 ,十 天 之 内 ,在 同一 台 PC 上 打开 淘宝 网 站 ,会 发 现 用 户 已 经 自动 登录 了 ,这 个 
功能 又 是 如 何 实现 的 呢 ? 

这 个 功能 是 通过 使 用 Cookie, 在 PC 上 保留 用 户 输入 的 账号 和 密码 一 段 时 间 来 实 
现 的 。 


: 在 登录 页 添加 用 户 选 择 是 否 保存 用 户 名 和 密码 ,并 且 选 择 保存 多 久 的 功能 。 
”技能 训练 : 
: 。 Cookie 的 使 用 。 

。 在 Servlet 中 处 理 Cookie。 
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8.3.1 Cookie 基础 


网 站 个 人 化 是 Cookie 最 有 益 的 用 途 之 一 。Cookie 浏览 某 网 站 时 ,网 站 存储 在 浏览 
器 所 在 的 客户 端 上 的 一 个 小 文本 文件 . 它 记录 了 用 户 ID, 密 码 ` 浏 览 过 的 网 页 .停留 的 时 
间 等 信息 , 当 同 一 主机 再 次 来 到 该 网 站 时 ,网 站 通过 读 取 Cookie, 得 知 该 主机 的 相关 信 
息 ,就 可 以 做 出 相应 的 动作 ,比如 不 用 输入 用 户 名 、 密 码 就 直接 登录 等 。 

用 户 使 用 某 台 主机 第 一 次 登录 网 站 ,输入 用 户 名 和 密码 后 ,用 户 名 和 密码 通过 
Cookie 保存 在 该 主机 上 , 当 用 户 下 一 次 通过 这 人 台 主 机 访问 这 个 网 站 的 时 候 , 可 以 直接 从 
本 地 Cookie 中 读 出 用 户 名 和 密码 ,不 用 再 重新 输入 了 。 

因为 需要 将 信息 保存 在 客户 端的 机 器 上 .所 以 ,Cookie 可 能 会 带 来 安全 性 问题 。 因 
此 ,项 目 开发 过 程 中 ,不 要 将 敏感 的 信息 (如 银行 账号 等 ) 保 存 到 Cookie 中 ,或 者 在 保存 这 
些 信息 之 前 提示 用 户 ,确认 信息 不 是 保存 在 公共 计算 机 上 。 

处 于 安全 性 考虑 ,浏览 器 可 以 被 设置 成 拒绝 Cookie, 所 以 项 目的 功能 实现 不 要 对 
Cookie 有 很 大 的 依赖 性 。 


8.3.2 编写 处 理 Cookie 的 类 


Cookie 的 实例 ,可 以 通过 Cookie 的 构造 器 来 创建 , 它 接收 两 个 String 类 型 参数 ,用 
于 指定 Cookie 的 属性 名 称 和 属性 值 。 

在 创建 好 Cookie 对 象 后 ,需要 将 它 发 送 到 客户 端 , 可 以 使 用 HttpServletResponse 的 
addCookie() 方 法 来 实现 , 它 接收 一 个 Cookie 类 型 的 值 ,并 将 这 个 Cookie 发 送 到 客户 端 。 
可 以 多 次 使 用 addCookie() 方 法 来 将 不 同 的 Cookie 发 送 到 客户 端 。 

可 以 通过 对 象 Cookie 的 setMaxAge() 方 法 来 设置 cookie 的 “生存 期 ”, 这 个 方法 接 
收 一 个 以 秒 为 单位 的 整数 参数 ,表示 这 个 cookie 可 以 在 客户 端的 保存 时 间 。 如 果 将 它 设 
置 成 0, 则 表示 在 客户 端 删除 了 这 个 Cookie。 

示例 8-9 按照 上 述 步骤 处 理 Cookie 的 源 代码 ,在 这 个 程序 中 ,首先 创建 了 两 个 
Cookie 对 象 ,分 别 用 来 储存 表单 中 传递 过 来 的 用 户 名 和 密码 ,然后 根据 客户 端 传递 的 
saveCookie 参数 ,决定 是 向 客户 端 发 送 cookie, 还 是 删除 以 前 存储 的 cookie, 通 过 Cookie 
对 象 的 setMaxAge() 可 以 实现 这 个 目的 。 

闪 示 例 8-10 ”Cookie 处 理 类 CookieUtil. java 

Package com.jyw.www.util; 

import javax.servlet.http.Cookie; 

import javax.servlet.http.HttpServletRequest; 


import javax.servlet .http.HttpServletResponse; 


public class CookieUtil { 
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public static void addCookies (HttpServletRequest request, 
HttpServletResponse response){ 


String account=request .getParameter ("account"); 
String password=request .getParameter ("password"); 获取 账号 和 密码 


Cookie userCookie=new Cookie("account", account); 》 


创建 Cookie 
一 的 新 实例 


Cookie pwdCookie=new Cookie ("pwd", password); 


if (request.getParameter ("saveCookie") !=null 
&& request .getParameter ("saveCookie") .equals ("Yes")) { 


如 果 用 户 要 求 记 住 用 户 名 和 密码 


userCookie.setMaxAge (7* 24* 60* 60) - - 
pwdCookie.setMaxAge (7* 24* 60* 60); 》 设置 Cookie 生 存 期 为 天 
} else { 


userCookie.setMaxAge (0) 如 果 用 户 没有 此 求 记 住 用 户 名 
pwdCookie.setMaxAge (0) } 和 密码 ， 则 删除 Cookie 


} 


response.addCookie (userCookie); 将 Cookie 发 送 到 客户 应 
送 到 
response.addCookie (pwdCookie); 2 


8.3.3 在 servlet 中 调用 Cookie 处 理 类 


用 户 输入 的 用 户 名 和 密码 是 被 Servlet 接收 并 进行 下 一 步 处 理 的 ,所 以 需要 继续 改写 
示例 8-8 中 的 doPost 方法 ,在 其 中 调用 8. 3. 2 小 节 编 写 的 Cookie 处 理 类 CookieUtil. 


java。 


闪 示 例 8-11 LoginServlet. java 


public void doPost (HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException { 
request.setCharacterEncoding ("GB18030"); 
response.setContentType ("text/html;charset=GB18030"); 


CookieUtil.addCookies (request, response); 


调用 实例 8-10 中 的 Cookie 处 理 方法 ， 按 
一 照 用 户 要 求 ， 保 存 用 户 的 账号 和 密码 


EmployeeDAO employeeDao=new EmployeeDAO () 

HttpSession sess=request.getSession (true); 

Employee employee=new Employee (); 

String account=request .getParameter ("account"); 

employee.setAccount (account); 

String password=request .getParameter ("password"); 

employee.setPassword (password); 

Integer authority=Integer.parselInt (request .getParameter ("authority")); 
employee.setAuthority (authority); 
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String code= (String) sess.getAttribute ("code"); 
String valCode=request .getParameter ("valCode"); 
try{ 
if (valCode !=null && valCode .equals (code)) { 
if (employeeDao .isEmployee (employee)) { 
Switch (authority) { 
case 0: 
request.getRequestDispatcher ("/showSales.jsp") .forward 
(request，response) ;break; 
case 1: 
request.getRequestDispatcher ("/showOrders .jsp") .forward 
(request, rsponse);break; 
case 2: 
request .getRequestDispatcher ("/searchStock.jsp") .forward 
(request, response); 
} 
} else { 
request .getRequestDispatcher ("/error.jsp") .forward (request, 
response) 


}elsel{ 
PrintWriter out=response.getWriter (); 
out .print ("验证 码 错 误 "); 
out .print("<a href='/login.jsp'>"+" 回 登录 页 "+"</a>"); 
out.flush(); 
out.close(); 

} 

} catch (Exception e) { 

e.printStackTrace (); 


8.3.4 改写 登录 页 面 


在 将 Cookie 发 送 到 客户 端 以 后 ,可 以 使 用 HttpServletRequest 的 getCookies() 方 法 
从 客户 端 获得 这 个 网 站 所 有 的 Cookie, 它 返回 包含 所 有 本 站 Cookie 的 数组 ,然后 通过 遍 
历 这 个 数组 就 可 以 获得 所 需 的 Cookie。 

在 login. jsp 中 加 入 上 述 处 理 ,并 且 添加 一 个 列表 菜单 ,让 用 户 可 以 选择 是 否 记 住 用 
户 名 和 密码 。 

闪 示 例 8-12 login. jsp 


<%Q@page contentTYPe= "text/html;charset=gb18030" language="java" import="java. 
EL 
<html> 

<head><title> 登 录 页 < /title></head> 
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<body align="center"> 


<jsp:include page= "menu.jsp"/> 


<% 
Cookie[] mycookies=null; 
int i; 
String account=null; 
String password=null; 
人 (Geauest .getCookies() pm (了 站 所 有 的 cookie ] 
mycookies=request .getCookies (); 
for (i=0; i<mycookies.length; i++) | 遍历 所 有 的 Cookie 
if (mycookies [1i] .getName () .equals ("account")) { 
account=mycookies[i] .getValue ();. 
Jelse if (mycookies [i] .getName () .equals ("pwd")) { 
password=mycookies [i] .getValue() 和 找 出 存储 在 Cookie 中 的 密码 | 
} 
} 
; 
多 > 


<form name="forml" method="post" action="/jyw/checklogin"> 
用 户 名 <INPUT type="text" name="username" 
value='<%= (account!=null)?account:""%$>'> 


将 账号 显示 在 用 于 
输入 用 户 名 的 元 素 中 


密 ” 码 <input type="password" name="password" 
value='<%= (password!=null)?password:""%>'> 
请 选择 身份 :<select name="authority"> 
<option value="0"> 管 理 员 < /option> 
<option value="1"> 总 店 店 员 < /option> 
<option value="2"> 分 店 店员 < /option> 
</select> 


将 密码 显示 在 用 于 
输入 密码 的 元 素 中 


选择 <select name="saveCookie"> 
<option value="No"> 不 记 账 号 和 密码 < /option> 
<option value="Yes"> 记 住 账号 和 密码 < /option> 
</select> 
<input type="submit" value=" 登 录 "/> 
</p> 
</form> 
<body> 
<html> 


修改 后 的 登录 页 执行 效果 如 图 8-8 所 示 , 记 住 用 户 名 和 密码 的 任务 就 完成 了 。 
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-jsjx| 
SO- |B hwp/ocahostsoe0noainjsp 7| [llx) [Esa-F. sa [5 


突 帘 名 hupy/ocalhostsosologinj-| | 价 “ 园 一 几 “ 四 mp ”好 RO) 


验证 码 as 7493 
请 选择 身份 : [3EERG 
选择 ees 可 

Ea 


图 8-8 ”可 以 记 住 用 户 账号 和 密码 的 登录 页 


8.4 任务 四 : 用 户 密码 的 MDS 加 密 


各 分 店 的 店员 应 能 根据 所 分 配 的 账号 登录 系统 进行 商品 库存 或 销售 信息 的 查询 、 更 
改 等 操作 。 这 就 需要 事先 为 员工 分 配 账号 等 信息 ,并 设置 初始 密码 。 这 个 功能 可 以 用 第 
5 章 学 到 的 数据 表 记 录 的 插入 技术 来 完成 。 但 是 本 章 对 这 一 功能 的 实现 提出 了 新 的 要 
求 : 为 了 安全 起 见 , 需 要 为 用 户 密码 加 密 。 


| 添加 各 店员 账号 的 初始 化 录入 功能 。 
。 要 求 输入 店员 的 账号 、 密 码 、 真 实 姓名 ,选择 其 所 属 分 店 , 单 击 “提交 ”按钮 后 ， 
存储 到 员工 信息 表 中 。 : 
。 对 用 户 密码 进行 加 密 存储 , 即 在 数据 表 中 看 到 的 用 户 密码 是 经 过 加 密 处 理 后 ; 
: 的 密 文 。 : 
”技能 训练 : 
| 。 熟练 掌握 向 数据 表 添加 一 条 信息 的 实现 技术 。 
。 MD5 加 密 算法 的 编写 。 


8.4.1 店员 基本 信息 录入 页 面 的 编写 


示例 8-13 和 示例 8-14 按照 任务 要 求 给 出 员工 信息 录入 功能 的 前 台 处 理 页 ,没有 使 
用 新 的 技能 ,读者 可 以 当 作 巩 固 性 的 练习 来 完成 编写 。 
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闪 示 例 8-13 employeeReg. jsp 员工 信息 录入 表单 。 


<%se@page language="java" import="java.util.*" contentType="text/html; 
charset=gb18030"%><center> 
填写 注册 信息 
<formname="forml" method="post" action="addEmployee.jsp"> 
用 户 名 
<input type="text" name="account"> 
<br> 
密码 
<input type="password" name= "password"> 
<br> 
姓名 
<input type="text" name="name"> 
<br> 
所 属 分 店 
<select name="storenum"> 
<option value="0"> 
总 店 
</option> 
<option value="1"> 
一 分 店 
</option> 
<option value="2"> 
二 分 店 
</option> 
</select> 
<br> 
授权 
<select name="authority"> 
<option value="0"> 
管理 员 
</option> 
<option value="1"> 
总 店 店员 
</option> 
<option value="2"> 
分 店 店员 
</option> 
</select> 
<br> 
<input type="submit" name="submit" value=" 添 加 " /> 
<input type="reset" name="reset" value=" 重 置 " /> 
</form> 
</center> 


执行 效果 如 图 8-9 所 示 。 
闪 示 例 8-14 addEmployee. jsp 


<se@epage language="java" import="java.util.* ,com.jyw.www.dao.*™ 
contentType="text/html;charset=gb18030"%> 
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EE 


8-9 员工 信息 录入 页 


<% request.setCharacterEncoding ("GB18030"); 
response.setContentType ("text/html;charset=GB18030"); 
多 > 
<jsp:useBean id="employee" class="com.jyw.www.entity.Employee" scope="page"/> 
<jsp:setProperty name= "employee" property="*" /> 
<% 
EmployeeDAO eDao=new EmployeeDAO(); 
eDao .save (employee); 


和 > 


8.4.2 编写 对 密码 进行 MD5 加 密 的 类 


用 户 密码 以 明文 的 方式 存储 在 数据 表 中 是 不 安全 的 ,一 般 都 需要 经 过 加 密 后 再 存储 。 


MD5 算法 是 一 种 不 可 逆 的 数据 加 密 算 法 , 即 在 获得 密 文 的 情况 下 也 无 法 还 原 出 明文 ,有 
较 好 的 安全 性 ,被 许多 Web 应 用 所 采用 。 下 面 编写 一 个 用 于 MD5 加 密 的 类 和 方法 ,将 其 
放 在 项 目的 util 包 中 。 
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闪 示 例 8-15 MD5Hashing 类 中 getMD5Hashing 方法 的 编写 


package com.jyw.www.util; 


import java.security.MessageDigest; 
import java.security.NoSuchAlgorithmException; 


public class MD5Hashing 
{ 
public String getMD5Hashing (String password) throws NoSuchAlgorithmException 
MessageDigest md; 
md=MessageDigest.getInstance ("MD5"); 
md.update (password.getBytes ()); 


byte byteData[]=md.digest (); 


第 8 章 于 各 守信 瓜 制 | 


StringBuffer sb=new StringBuffer(); 
for (int i=0; i<byteData.length; i++) { 
sb.append (Integer.toString ((byteData[i]&0xff)+0x100,16) .substring (1)); 


StringBuffer hexString=new StringBuffer (); 

for (int i=0;i<byteData.length;i++) { 
String hex=Integer.toHexString (Oxff & byteData[i]); 
if (hex.length()==1) hexString.append('0'); 
hexString.append (hex); 

} 

return hexString.tostring(); 


8.4.3 编写 EmployeeDAO 类 中 的 save 方法 


DAO 类 中 save( ) 方 法 的 编写 规则 ,在 第 5 章 中 已 经 给 出 了 示例 ,完成 此 任务 的 不 同 
之 处 在 于 在 save() 方 法 中 需要 调用 示例 8-15 中 的 getMD5 Hashing 方法 ,获得 密 文 返回 
值 后 ,将 密 文 写 人 相应 数据 表 中 。 

闪 示 例 8-16 ”EmployeeDAO 类 中 save() 方 法 


public void save (Employee employee) { 
try { 
InitialContext ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds .getConnection ()? 
PreparedStatement pstmt=conn.prepareStatement ( 
"INSERT INTO employee 
(account,password, storenum,name,previlige) VALUES (?3,23,3,?3,3)", 
ResultSet .TYPE SCROLL INSENSITIVE, 
ResultSet .CONCUR READ ONLY); 
pstmt.setString(1, employee.getAccount ()); 
String password=employee.getPassword(); 


MD5Hashing mh=new MD5Hashing (); 将 加 密 后 的 密码 
写 人 数据 表 上 


车 


Pstmt .setString (2, mh.getMD5Hashing (password)); 
pstmt .setInt (3, employee.getStorenum()); 
Pstmt .setString(4，employee.getName ()); 
Pstmt .setInt (5, employee.getAuthority()); 
Pstmt .executeUpdate (); 
Pstmt .close() 
} catch (Exception e) { 
e.printstackTrace (); 
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录入 新 的 员工 信息 后 ,打开 employee 数据 表 . 可 以 看 到 在 password 列 中 的 密码 都 是 
以 密 文 的 方式 存储 的 ,看 不 出 真实 的 密码 ,如 图 8-10 所 示 。 
用 户 输入 的 密码 在 数据 库 中 以 密 文 的 形式 存储 


el0adc3949ba59abbe56e057f207863e 
el0adc3949ba59abbe56e057f20f663e 


el0adc3949ba59abbe56e057f20f683e 


(NULL) 


图 8-10 经 md5 加 密 后 对 用 户 密码 的 示意 


8.4.4 改写 EmployeeDAO 类 中 的 isEmployee 方法 


完成 8. 4. 3 小 节 后 ,用 户 在 登录 时 ,即便 输入 的 账号 和 密码 正确 也 不 能 正常 登录 ,这 
是 因为 用 户 登 录 时 输入 的 是 明文 的 密码 ,而 数据 表 中 的 密码 以 密 文 的 形式 存储 。 要 恢复 
登录 功能 就 需要 把 用 户 输入 的 密码 以 同样 的 算法 加 密 ,在 与 数据 表 中 的 密 文 相 比 较 来 判 
断 输入 的 密码 是 否 正 确 。 所 以 ,本 章 任务 一 中 示例 8-2 的 isEmployee() 方 法 要 进行 修改 
如 下 。 

闪 示 例 8-17 EmployeeDAO 类 中 的 isSEmployee() 方 法 


public Boolean isEmployee (Employee employee) throws Exception { 
InitialContext ctx=new InitialContext (); 
DataSource ds= (DataSource) ctx.lookup ("java:comp/env/jdbc/mysql"); 
Connection conn=ds.getConnection(); 
PreparedStatement pstmt=conn.prepareStatement ( 
"SELECT * FROM employee WHERE account=?and 
password=?and authority=?"， 
ResultSet .TYPE_SCROLL_INSENSITIVE， 
ResultSet .CONCUR READ ONLY); 
pstmt.setString(1, employee.getAccount ()); 


以 加 密 后 的 密码 
作为 查询 条 件 


String password=employee .getPassword() 
MD5Hashing mh=new MD5Hashing() : 


pstmt.setSstring(2, mh.getMDSHashing (password)); 


Pstmt .setInt (3, employee.getAuthority()); 
ResultSet rs=pstmt .executeQuery (); 
if (rs.first()) { 
return true; 
} else { 


return false; 


} 


修改 完成 后 ,登录 功能 就 可 以 恢复 了 。 
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8.5 ”任务 五 : 已 登录 用 户 的 身份 跟踪 


问题 : 用 户 一 次 登录 Web 应 用 系统 后 ,无 论 切 换 到 哪个 页 面 ,都 可 以 被 识别 ; 某 些 
Web 应 用 系统 ,如 淘宝 网 ,还 会 将 用 户 名 一 直 显 示 在 “页 眉 ” 处 ; 当 用 户 离开 计算 机 较 长 时 
间 后 ,发 现 已 经 自动 退出 了 系统 。 这 是 如 何 实现 的 呢 ? 

”任务 目标 : 
| 用 户 登 录 “ 佳 衣 屋 ”系统 后 ,在 任何 一 个 网 页 均 可 识别 用 户 身 份 ,显示 “欢迎 X X X” 
| 且 和 给 出 “退出 ” 超 链接 ,供用 户 使 用 完 系统 功能 后 ,退出 系统 之 用 。 用 户 单 击 
“退出 ” 超 链接 后 ,欢迎 信息 和 "退出 ” 超 链 接 不 再 出 现 。 : 
技能 训练: 
。 Servlet 处 理会 话 追 踪 。 
。 Session 常用 方法 的 使 用 。 


8.5.1 Session 简介 


本 节 中 ,采用 Session( 会 话 ) 技 术 来 完成 任务 五 。 

Session 的 基本 思路 是 : 给 每 个 客户 端 分 配 一 个 不 重复 的 ID 来 区 分 不 同 的 客户 端 ， 
而 将 对 应 各 个 客户 需要 保存 的 数据 保存 在 服务 器 的 内 存 中 ,因为 不 需要 写 到 客户 端的 文 
本 中 ,所 以 ,在 服务 器 中 可 以 存放 任何 的 对 象 而 不 单纯 是 String 数据 。 

Session 封装 在 javax. servlet. http. HttpSession 这 个 接口 中 ,这 个 接口 构建 在 Cookie 或 
者 URL 重 写 的 基础 上 。 要 得 到 一 个 HttpSession 的 实例 ,可 以 通过 HttpServletRequest 的 
getSession() 方 法 来 获得 , HttpServletRequest 有 两 个 重 载 的 getSession() 方 法 ,一 个 不 带 任何 
参数 ,一 个 接收 boolean 类 型 的 值 。getSession() 方 法 和 getSession(true) 功 能 一 样 ,如 果 对 应 
的 客户 端 已 经 产生 过 一 个 session, 那 么 就 会 返回 这 个 旧 的 session, 和 否则 ,这 个 方法 将 产生 一 
个 session ID 并 且 和 对 应 的 客户 端 绑 定 在 一 起 。 而 如 果 getSession(false) 表 示 对 应 的 客户 端 
已 经 有 session, 那么 返回 这 个 旧 的 session, 否则 ,不 会 产生 新 的 session。 可 以 使 用 
HttpSession 对 象 上 的 isNew() 方 法 来 判定 这 个 session 是 否 为 新 建 的 。 

使 用 Session 技术 完成 任务 五 的 要 点 有 以 下 四 点 。 

(1) 已 登录 用 户 的 用 户 名 需要 保存 在 Session 中 。 使 用 Session 对 象 的 void 
setAttribute(String attrName,Object obj ) 方 法 。 

(2) 设置 Session 处 于 连续 非 活 跃 状态 的 生存 期 。 使 用 Session 对 象 的 void 
setMaxInactiveInterval(Integer seconds) 方 法 。 

(3) 在 需要 使 用 用 户 名 的 地 方 ,从 Session 中 取出 。 使 用 Session 对 象 的 Object 
getAttribute(String attrName) 方 法 。 
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退出 ? 超 链 接 后 使 Session 失效 。 使 用 Session 对 象 的 void invalidate() 


(4) 用 户 单刀 
方法 。 


8.5.2 Servlet 中 使 用 Session 


首先 来 改写 LoginServlet. java 中 的 doPost 方法 ,将 用 户 已 登录 的 状态 信息 保存 到 
Session 中 。 
闪 示 例 8-18 LoginServlet. java 


public void doPost (HttpServletRequest request, HttpServletResponse response) 
throws ServletException, IOException { 
request.setCharacterEncoding ("GB18030"); 
response.setContentType ("text/html;charset=GB18030"); 
CookieUtil.addCookies (request, response); 
EmployeeDAO employeeDao=new EmployeeDAO (); 
HttpSession sess=request.getSession(true); 
Employee employee=new Employee(); 
String account=request .getParameter ("account"); 
employee.setAccount (account); 
String password=request .getParameter ("password"); 
employee.setPassword (password); 
Integer authority=Integer.parseInt (request .getParameter ("authority")); 
employee.setAuthority (authority); 
String code= (String) sess.getAttribute ("code"); 
String valCode=request .getParameter ("valCode"); // 获取 用 户 输入 的 验证 码 
try { 
if (valCode !=null && valCode.equals (code)) { 
if (employeeDao.isEmployee (employee)) { 
request.getSession() .setAttribute ("account", account); 


把 用 户 登 录 时 输入 的 用 户 名 保存 在 Session 


request.getSession() .setAttribute ("authority", authority); 


Ht 


把 用 户 登 录 时 选择 的 角色 保存 在 Session 中 
request.getSession() .setMaxInactiveInterVal (10* 60) 
switch (authority) { 
case 0: [设置 Session 非 活跃 状态 的 生存 期 为 10 分 名 
request .getRequestDispatcher("/showSales.jsp") .forward!( 
request, response); 
break; 
case 1: 
request .getRequestDispatcher ("/showOrders.jsp") 
.forward (request, response); 
break; 
case 2: 
request .getRequestDispatcher("/searchStock.jsp") .forward( 


request, response); 
上 
} else { 
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request.getRequestDispatcher("/error.jsp") .forward (request, 
response); 
} 
}else{ 
PrintWriter out=response.getWriter (); 
out .print ("验证 码 错误 "); 
out.print ("<a href='/login.jsp'>"+" 回 登录 页 "+"</a>"); 
out.flush(); 
out.close(); 
} 
} catch (Exception e) { 
e.printStackTrace (); 


8.5.3 编写 显示 欢迎 信息 页 眉 的 JSP 页 面 


本 小 节 编 写 一 个 简单 header 页 面 来 示例 如 何 从 Session 中 取出 一 个 属性 的 值 。 
闪 示 例 8-19 header.jsp 


<% 
String account= (String) session.getAttribute ("account"); 


if (account !=null) { 


从 Session 中 取出 用 户 账 号 对 象 ， 并 强制 转换 为 字符 串 


out .Print (account) 7 
%> 已 登录 

<a href="logout.jsp"> 退 出 </a> 
<% 


和 > 


用 户 chenf 登录 后 的 执行 效果 引用 了 header.jsp 的 页 面 执行 效果 ,如 图 8-11 所 示 。 


Rn 可 


TO + [Brrp/ocanerraanchectcon 
齐集 由 hrpyrlocalhostaosoicheckoogh | | 从 ~ 加- 大- 世 FEPv 切 IRO > 


chenf 已 登录 退出 


库存 查询 结果 
商品 编号 [ 商品 名 称 [ 颜色 [ 尺码 六 司 库存 量 | 梧 于 吉 


JWIZCKNI JWI2CKY2 JWl2CKNI 
和 牛仔 祥 牛仔 祥 | 后 仔 祥 
佳 衣 悍 总 店 华强 北 店 || 佳 衣 尾 总 店 华强 北 店 ||， 佳 衣 悍 海岸 域 店 
MM 水 洗 黑 [入 123. 0|33||M 院 | 广 158.0 |30 | ¥123. 0[20 


El 


图 8-11 带 欢 迎 信息 和 “退出 ” 超 链接 的 页 面 
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8.5.4 编写 实现 退出 系统 的 JSP 页 


由 于 Session 中 “存储 ”了 用 户 登 录 的 信息 ,所 以 如 果 使 Session“ 失 效 ”, 则 用 户 已 登录 
的 信息 就 不 复 存 在 了 ,按照 这 一 思路 ,编写 一 个 简单 的 退出 页 面 来 示例 如 何 使 Session 
失效 。 

闪 示 例 8-20 logout. jsp 

<% 


session,.invalidate (); 使 Session 失 效 ， 完 成 退出 


response.sendRedirect ("login.jsp"); 
多 > 


用 户 单 击 header. jsp 中 的 “退出 ? 超 链 接 ,将 调用 示例 8-19 的 logout. jsp ,使 Session 
失效 。 


8.6 任务 六 : 角色 权限 的 过 波 


到 目前 为 止 ,为 佳 衣 屋 ” 系 统 添加 了 许多 功能 模块 ,这 些 功 能 无 论 用 户 是 否 登 录 都 可 
以 使 用 ,甚至 添加 系统 用 户 这 类 权限 的 管理 员 才 有 权 操 作 的 功能 也 是 开放 的 ,这 显然 不 符 
合 安 全 性 的 要 求 , 任 务 六 就 要 求 为 佳 衣 屋 ”系统 按 角 色 分 配 操 作 权 限 。 


| 按照 指定 的 系统 结构 图 ,对 系统 用 户 按 角色 进行 权限 划分 ,例如 : 只 有 管理 员 登 : 
; 录 后 才能 实现 员工 信息 录入 的 功能 。 
| 要求: 
。 将 受 控 文 件 放 到 相应 的 目录 中 。 
。 参考 练习 10. 3 可 以 为 项 目 添加 管理 员 登 录 filter 过 滤器 。 
。 正确 配置 Web. xml 文件 。 
”技能 训练 : 
: 。 Filter 的 编写 。 
。 Filter 的 配置 。 


8.6.1 Filter 过 滤器 简介 


过 滤器 是 一 种 小 型 的 ,可 插入 的 Web 组 件 , 其 提供 了 对 Web 应 用 程序 的 前 期 处 理 和 
后 期 处 理 的 逻辑 控制 ,可 以 拦截 请 求 和 响应 ,以 便 查看 、 提 取 或 以 某 种 方式 操作 正在 客户 
端 和 服务 器 之 间 进行 交换 的 数据 。 与 Servlet 类 似 , 过 滤器 也 需要 在 Web 应 用 配置 文件 
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( 即 Web. xml) 中 进行 配置 部 署 。 

过 滤器 并 不 是 Servlet, 过 滤器 并 不 实际 创建 一 个 请 求 。 而 是 请 求 到 达 一 个 Servlet 
前 的 预 处 理 程 序 或 响应 离开 Servlet 后 的 后 处 理 程序 。 一 个 过 滤器 能 够 实现 如 下 功能 : 

。 在 一 个 Servlet 被 调用 前 截获 该 调用 。 

。 在 一 个 Servlet 被 调用 前 检查 请 求 。 

。 修改 在 实际 请 求 中 提供 了 可 定制 请 求 对 象 的 请 求 头 和 请 求 数据 。 

。 修改 在 实际 响应 中 提供 了 可 定制 响应 对 象 的 响应 头 和 响应 数据 。 


8.6.2 使 用 过 滤器 实现 访问 权限 控制 


(1) 将 不 同 访问 权限 的 页 面 置 于 不 同 的 目录 下 。 比 如 : 将 只 允许 管理 员 可 见 的 内 容 
都 放 在 根 目录 的 admin 目录 下 。 

(2) 编写 过 滤器 类 。 

创建 一 个 类 ,实现 Filter 接口 ,并 且 实 现 其 中 的 init() doFilter() 和 destroy( ) 方 法 。 
doFilter() 方 法 是 Filter 类 的 核心 方法 ,希望 过 滤器 完成 的 功能 ,都 应 该 放 到 这 个 方法 中 。 

doFilter() 方 法 有 3 个 参数 : ServletRequest、ServletResponse 和 FilterChain, 其 中 
ServletRequest 和 ServletResponse 为 传递 给 方法 的 请 求 和 响应 参数 ,而 FilterChain 可 以 
用 来 把 请 求 和 响应 传递 给 下 一 个 Filter 或 者 其 他 JSP/Servlet 等 资源 。 

在 doFilter() 中 调用 FilterChain 的 doFilter() 方 法 , 它 只 有 两 个 参数 : ServletRequest 和 
ServletResponse ,通常 只 要 将 Filter 的 doFilter( ) 方 法 的 前 两 个 参数 当 作 它 的 参数 就 可 
以 下 < 

闪 示 例 8-21 实现 管理 员 权 限 控制 的 过 滤器 AdminFilter. java 


Package com.jyw.www.filter; 
import java.io.IOException; 


import javax.servlet .Filter7 

import javax.servlet .FilterChainy” 

import javax.servlet.FilterConfig; 

import javax.servlet.ServletExceptiony7 

import javax.servlet .ServletRequest; 

import javax.servlet .ServletResponse; 

import javax.servlet.http.HttpServletRequest; 


import javax.servlet.http.HttpServletResponse; 


public void doFilter (ServletRequest request, ServletResponse response, FilterChain 
chain) 
throws IOException, ServletException { 
request.setCharacterEncoding ("gb18030"); 
response.setCharacterEncoding ("gb18030"); 
HttpServletRequest httpreq= (HttpServletRequest)request; 
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HttpServletResponse httpresp= (HttpServletResponse)response; 
Object obj=httpreq.getSession() .getAttribute ("authority"); 


if (obj!=null) S 
从 Session 中 取出 用 户 角色 ， 如 果 没 有 ， 则 说 明 使 用 者 没有 登录 


Integer authority= (Integer)obj; 


jf (authority1-0) {一 管理 员 的 权限 值 是 9， 如果 不 是 9， 则 说 明 当 前 用 户 不 是 管理 员 


httpreq.getRequestDispatcher ("/privError.jsp") .forward (httpreq, httpresp); 


Jelsef{ 跳 转 到 权限 错误 提示 页 
chain.doFilter (httpreq, httpresp); 


) 如 果 是 管理 员 ， 则 转向 下 一 个 过 滤器 链 上 的 下 一 个 对 象 ， 
jelse | 如果 没有 下 一 个 对 象 ， 则 通过 过 滤 ， 跳 转 到 请 求 访问 的 页 面 


httpreq.getRequestDispatcher ("/PrivError.jsp") .forward (httpreq, httpresp); 


} 

public void init (FilterConfig arg0) throws ServletException { 
// TODO Auto- generated method stub 

} 

public void destroy() { 
// TODO Auto- generated method stub 

} 


读者 可 以 参考 示例 8-20, 完 成 总 店 店员 过 滤器 HeadFilter. java 和 分 店 店员 过 滤器 


BranchFilter 的 编写 。 


8.6.3 在 Web.xml 中 配置 过 滤器 
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同 Servlet 的 用 法 相似 ,要 使 用 Filter, 也 需要 在 Web. xml 中 进行 配置 。 
闪 示 例 8-22 ”Web. xml 中 Filter 的 配置 


<filter> 
<filter-name>AdminFilter</filter-name> 
<filter-class>com.jyw.www.filter.AdminFilter</filter-class> 
</filter> 
<filter-mapping> 
<filter-name>AdminFilter</filter-name> 
<url-pattern>/admin/* </url-pattern> 


ts 设置 对 /admin 目 录 下 的 所 有 文件 进行 “过 滤 ” 

<filter> 该 目录 下 的 义 件 只 能 由 管理 员 访 问 
<filter-name>HeadFilter</filter-name> 
<filter-class>com.jyw.www.filter.HeadFilter</filter-class> 

</filter> 

<filter-mapping> 
<filter-name>HeadFilter</filter-name> 
<url-pattern>/employee/* </url-pattern> 

</filter-mapping> 


设置 对 /employee 目 录 下 的 所 有 文件 进行 "过滤 ”， 
该 目录 下 的 文件 只 能 由 总 店 店员 访问 


<filter> 
<filter-name>BranchFilter</filter-name> 
<filter-class>com.jyw.www.filter.BranchFilter</filter-class> 
</filter> 
<filter-mapping> 
<filter-name>BranchFilter</filter-name> 
<url-pattern> /employee/branch/* </url-pattern> 


0 设置 对 /employee/branch 目 录 下 的 所 有 文件 进行 “过滤 " 
该 目录 下 的 文件 只 能 由 分 店 店员 访问 ， 因 为 该 目录 为 
lemployee 的 子 目录 ， 所 以 总 店 店员 可 访问 该 自 录 下 的 文件 
现在 ,只 需要 将 JSP 文件 分 别 放 到 相应 的 目录 下 ,就 可 实现 不 同 角 色 的 权限 控制 了 。 
例如 ,将 实现 将 员工 录入 功能 的 addEmployeeForm. jsp 文件 移 至 根 目录 下 的 admin 子 目 
录 中 ,只 要 不 是 管理 员 登 录 , 对 其 进行 访问 时 都 会 跳 转 到 权限 错误 提示 页 ,如 图 8-12 
所 示 。 


SO -|B r/ochostaoe0/sdmin/addemployeeFormjsp 了 人 下 区 一 下 , 你 就 尖 道 


帘 安 ”条 htpy/localhostsoso/admiyaddEmployeeFo- 丛 ~ 园 - 蜗 ~ 辐 7Epv 二 > 
无 访问 权限 返回 登录 页 


图 8-12 访问 权限 错误 时 的 提示 信息 


8.7 知识 扩展 


8.7.1 Servlet 生命 周期 


Servlet 的 生命 周期 是 指 一 个 Servlet 从 创建 到 结束 的 一 系列 步骤 ,分 别 由 Servlet 中 
相应 的 方法 来 代表 。 


1. init 方法 


在 服务 器 上 最 多 只 会 驻 留 一 个 Servlet 的 实例 ,在 第 一 次 调用 这 个 Servlet 的 时 候 , 将 
会 创建 这 个 Servlet 的 实例 。 在 创建 这 个 Servlet 实例 的 时 候 , HttpServlet 中 的 init() 方 
法 会 被 调用 ,如 果 需 要 对 这 个 Servlet 做 一 些 初始 化 .可 以 将 初始 化 代码 放 在 这 个 方法 中 。 
注意 ,这 个 方法 只 会 被 调用 一 次 ,不 会 对 于 每 次 的 连接 都 调用 它 。 


2. Service 方法 


前 面 已 经 介绍 了 doGet() 和 doPost() 方 法 ,对 于 不 同方 式 的 请 求 , 会 白 动 去 调用 对 应 

的 方法 。 其 实 , 如 果 客 户 端 有 一 个 对 Servlet 的 请 求 发 送 过 来 ,那么 ,服务 器 会 产生 一 个 新 
的 线程 ,并 且 让 它 调用 Servlet 的 Service() 方 法 。Service() 方 法 会 根据 收 到 的 客户 端的 
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请 求 类 型 ,决定 调用 doGet() 还 是 doPost() 甚 至 其 他 的 doXXX() 方 法 。 
3. doGet /doPost 方法 


通过 覆盖 HttpServlet 类 中 的 doGet() 方 法 ,可 以 处 理 浏览 器 端 发 送 过 来 的 GET 请 
求 。 而 doPost() 方 法 用 于 处 理 POST 请 求 。 

这 两 个 方法 的 参数 一 样 : HttpServletRequest 和 HttpServletResponse, 它 们 也 是 分 
别 代表 客户 端的 请 求 和 对 客户 端的 响应 。 如 果 不 能 确定 客户 端的 请 求 方式 到 底 是 GET 
方式 还 是 POST 方式 ,那么 需要 在 Servlet 中 同时 定义 这 两 个 方法 。 如 果 这 两 个 方法 的 
处 理 过 程 一 样 ,那么 只 需要 定义 其 中 的 一 个 ,而 让 另外 一 个 调用 这 个 方法 。 


4. destroy 方法 


如 果 因 为 某 种 原因 ,需要 删除 某 个 Servlet 实例 ,那么 ,在 删除 这 个 Servlet 实例 之 前 ， 
服务 器 会 首先 调用 destroy() 方 法 。 可 以 在 这 个 方法 中 执行 一 些 清理 的 动作 ,比如 释放 数 
据 库 连 接 、 关 闭 打 开 的 文件 等 。 


8.7.2 Filter 生命 周期 

Filter 生命 周期 是 指 一 个 Filter 从 创建 到 结束 的 一 系列 步骤 ,分 别 由 Filter 中 相应 的 
方法 来 代表 。 

1. init 方法 

2. doFilter ”方法 


执行 实际 的 过 滤 工 作 。 在 doFilter() 方 法 中 ,每 个 过 滤器 都 接受 当前 的 请 求 和 响应 ， 
而 FilterChain 包含 的 过 滤器 则 仍然 要 被 处 理 。 过 滤器 调用 chain. doFilter() 将 控制 权 传 
送 给 下 一 个 过 滤器 。 


3. destroy 方法 


服务 器 调用 destory( ) 方 法 结束 过 滤器 。 
8.7.3 Filter 链 


Java 中 的 Filter 并 不 是 一 个 标准 的 Servlet, 它 不 能 处 理 用 户 请 求 , 也 不 能 对 客户 端 
生成 响应 ,主要 用 于 对 HttpServletRequest 进行 预 处 理 , 也 可 以 对 HttpServletResponse 
进行 后 处 理 , 它 是 个 典型 的 处 理 链 。 

Filter 有 如 下 几 个 用 处 。 

。 在 HttpServletRequest 到 达 Servlet 之 前 ,拦截 客户 的 HttpServletRequest。 
176 


。 根据 需要 检查 HttpServletRequest, 也 可 以 修改 HttpServletRequest 头 和 数据 。 

。 在 HttpServletResponse 到 达 客 户 端 之 前 ,拦截 HttpServletResponse。 

。 根据 需要 检查 HttpServletResponse, 可 以 修改 HttpServletResponse 头 和 数据 。 

Filter 有 如 下 几 个 种 类 。 

。 用 户 授 权 的 Filter: Filter 负责 检查 用 户 请 求 ,根据 请 求 过 滤 用 户 非法 请 求 。 

。 日 志 Filter: 详细 记录 某 些 特殊 的 用 户 请 求 。 

。 负责 解码 的 Filter: 包括 对 非 标准 编码 的 请 求解 码 。 

。 能 改变 XML 内 容 的 XSLTFilter 等 。 

对 同样 的 过 滤 目 标 ,哪个 过 滤器 在 Web. xml 中 先 配 置 ,在 过 滤器 链 中 就 会 先 得 到 调 
用 ,在 Web. xml 中 还 可 以 配置 Filter 的 一 些 初始 化 参数 。 下 面 举例 说 明 Filter 链 的 
用 法 。 

开发 Java Web 项 目 时 ,可 能 会 出 现 中 文 乱码 问题 ,比如 本 章 前 面 描述 的 用 户 名 和 密 
码 的 显示 结果 可 能 会 出 现 乱 码 , 这 是 由 于 中 文 简体 编码 集 不 是 浏览 器 默认 的 编码 集 的 原 
因 , 这 就 需要 程序 明确 地 “告诉 ”用 户 的 浏览 器 应 该 用 什么 编码 方式 来 解析 接收 到 的 输 
出 流 。 

在 Servlet 中 ,可 以 通过 修改 Content-Type 完成 ,也 就 是 在 获得 输出 流 之 前 ,利用 
setContentType() 方 法 去 设置 Content-Type 报头 。 

注意 : setContentType() 方 法 必须 在 HttpServletResponse 的 getWriter() 方 法 获得 
输出 流 和 句柄 之 前 调用 。 

通过 下 面 这 条 语句 ,就 可 以 将 输出 的 编码 方式 设置 成 GB18030 编码 方式 。 


response.setContentType ("text/html;charset=GB18030"); 


但 是 ,如 果 向 客户 端 输出 的 中 文 是 从 表单 中 取出 来 的 ,还 是 会 出 现 乱码 的 问题 ,这 
个 问题 也 是 因为 编码 方式 不 统一 引起 的 。Servlet 在 读 取 表单 数据 的 时 候 , 默 认 采 用 
iso-8859-1 的 编码 方式 ,此 时 ,需要 将 Servlet 读 取 表 单数 据 的 编码 方式 也 设置 为 
GB18030 ,这 个 任务 可 以 由 HttpServletRequest 的 setCharacterEncoding() 方 法 来 完成 。 
在 读 取 表单 数据 之 前 ,调用 这 个 方法 来 将 编码 设置 成 GB18030 即 可 ,示例 如 下 : 


request .setCharacterEncoding ("GB18030"); 


下 面 编写 编码 格式 过 滤器 来 实现 输入 /输出 编码 格式 的 设置 。 
闪 示 例 8-23 ”编码 格式 过 滤器 EncodingFilter. java 


Package com.jyw.www.filter; 


import java.io.IOException; 

import javax.servlet .Filter; 

import javax.servlet .FilterChain; 
import javax.servlet .FilterConfig; 
import javax.servlet .ServletContext; 
import javax.servlet .ServletException; 


import javax.servlet .ServletRequest; 


人 ZX 
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import javax.servlet.ServletResponse; 


public class EncodingFilter implements Filter { 
private FilterConfig filterConfig=null; 
private String encoding=null; 
public void destroy() { 
encoding=null; 


public void doFilter (ServletRequest request, ServletResponse response, 
FilterChain chain) throws IOException, ServletException { 
String encoding=getEncoding () 7 
if (encoding==null) { 
encoding="GB18030"; 


) 设置 输入 / 
request.setCharacterEncoding (encoding); 》 输出 编码 
response.setContentType ("text/html ;charset=GB18030"); 2 格式 为 

chain.doFilter (request, response); 中 文 简体 


public void init (FilterConfig filterConfig) throws ServletException { 
this.filterConfig=filterConfig; 
this.encoding= i ; 


读 取 Web.xml 中 的 初始 化 参数 


private String getEncoding() { 
return this.encoding; 


} 


将 下 面 的 EncodingFilter 配置 写 在 其 他 Filter 之 间 , 就 可 以 实现 链 式 调用 。 读 者 可 
以 使 用 MyEclipse 的 Debug 功能 来 跟踪 验证 。 
闪 示 例 8-24 Web. xml 中 EncodingFilter 的 配置 


<filter> 
<filter-name>encodingFilter</filter-name> 
<filter-class>com.jyw.www.filter.EncodingFilter</filter-class> 
<init-param> 
<param-name>encoding< /param- name> 
<param-value>GB18030< /param- value> 
</init-param> 
</filter> 编码 集 初始 化 参数 为 GB18030 
<filter-mapping> 
<filter-name>encodingFilter</filter-name> 


<url-pattern>/*</url-pattern> 
</filter-mapping> 
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8.7.4 Cookie 属性 的 读 写 


在 JSP 中 可 以 通过 cookie. setXXX 方法 设置 各 种 属性 ,用 cookie. getXXX 方法 读 出 
Cookie 的 属性 , Cookie 的 主要 属性 及 其 方法 见 表 8-1。 


表 8-1 Cookie 读 写 各 种 属性 的 方法 
类 型 方 法 名 方法 解释 

String getComment() 返回 Cookie 中 注释 ,如 果 没 有 注释 将 返回 空 值 
返回 Cookie 适用 的 域名 。 使 用 getDomain() 方 法 可 以 指 

String 人 示 浏 览 器 把 Cookie 返回 给 同一 域内 的 其 他 服务 器 ,而 通 
常 Cookie 只 返回 给 发 送 它 的 服务 器 名 字 完 全 相同 的 服 
务 器 。 注 意 域名 必须 以 点 开始 (例如 . yesky. com) 

int getMaxAge() 返回 Cookie 过 期 之 前 的 最 大 时 间 ,以 秒 计 算 

String getName() 返回 Cookie 的 名 字 

String getPath() 返回 Dontie 适用 的 路 径 二 和 不 指 定 路 径 Re 将 返 
回 给 当前 页 面 所 在 日 录 及 其 子 日 录 下 的 所 有 页 面 

boolean getSecure() 各 第 浏览 烽 通 过 安全 协 次 发 送 Coe; 条 返回 os 
如 果 浏 览 器 使 用 标准 协议 , 则 返回 false 值 

String getValue() 返回 Cookie 的 值 

int getVersion() 返回 Cookie 所 遵从 的 协议 版 本 

void setComment(String purpose) 设置 Cookie 中 注释 

void setDomain(String pattern) 设置 Cookie 适用 的 域名 

void setMaxAge(int expiry) 以 秒 计算 ,设置 Cookie 过 期 的 时 间 

void setPath(String uri) 指定 Cookie 适用 的 路 径 

void setSecure( boolean flag) 指出 浏览 器 使 用 的 安全 协议 ,例如 HTTPS 或 SSL 

void setValue(String newValue) Cookie 创建 后 设置 一 个 新 的 值 

void setVersion(int v) 设置 Cookie 所 遵从 的 协议 版 本 


8.7.5 ”Cookie 的 生存 周期 


Cookie 可 以 保持 登录 信息 到 用 户 下 次 与 服务 器 的 会 话 , 换 名 话说 ,下 次 访问 同一 网 
站 时 ,用 户 会 发 现 不 必 输 入 用 户 名 和 密码 就 已 经 登录 了 (当然 ,不 排除 用 户 手 工 删 除 
Cookie) 。 而 还 有 一 些 Cookie 在 用 户 退出 会 话 的 时 候 就 被 删除 了 ,这 样 可 以 有 效 地 保护 


个 人 隐私 。 


Cookie 在 生成 时 就 会 被 指定 一 个 Expire 值 ,这 就 是 Cookie 的 生存 周期 ,在 这 个 周期 
内 Cookie 有 效 ,超出 周期 Cookie 就 会 被 清除 。 有 些 页 面 将 Cookie 的 生存 周期 设置 为 0 


或 负 值 , 这 村 


# 在 关闭 页 面 时 ,就 马上 清除 Cookie, 不 会 记录 用 户 信 息 ,使 网 页 更 加 安全 。 


在 JSP 中 ,使 用 setMaxAge(int expiry) 方 法 来 设置 Cookie 的 生存 周期 。 参 数 expiry 
应 是 一 个 整数 , 正 值 表示 Cookie 将 在 多 少 秒 以 后 失效 。 注 意 这 个 值 是 Cookie 将 要 存在 
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的 最 大 时 间 ,而 不 是 Cookie 现在 的 存在 时 间 。 负 值 表示 当 浏 览 器 关闭 时 ,Cookie 将 会 被 
删除 。 零 值 则 是 要 删除 该 Cookie。 


8.7.6 MVC 设计 模式 


在 本 章 之 前 ,项 目的 功能 模块 都 是 采用 JSP 十 JavaBean 的 模式 来 完成 的 ,通过 将 复杂 
的 程序 代码 封装 到 JavaBean 中 ,减少 了 JSP 代码 和 网 页 标签 混合 使 用 的 情况 ,在 阅读 和 
维护 的 时 候 会 带 来 很 大 的 方便 :同时 将 公用 的 代码 封装 在 JavaBean 中 ,也 提高 了 代码 的 
可 复 用 性 。 但 是 由 于 JSP 页 面 中 还 是 嵌入 了 较 多 的 Java 代码 , 当 需 要 处 理 的 业务 逻辑 非 
常 复 杂 时 ,大 量 的 Java 代码 使 得 JSP 页 面 变 得 非常 腔 肿 ,前 端的 页 面 设 计 人 员 稍 有 不 慎 ， 
就 有 可 能 破坏 有 关 商 业 逻 辑 的 代码 。 

本 章 引入 了 Servlet 技术 来 完成 了 用 户 登 录 等 模块 的 功能 ,采用 了 MVC, 即 Model- 
View-Controller 模式 。 在 这 种 模式 中 ,Servlet 充当 了 控制 器 (Controller 即 C) 的 角色 , 负 
责 响 应 客户 对 业务 逻辑 的 请 求 并 根据 用 户 的 请 求 行 为 决定 将 要 调用 的 JSP 页 面 。 

JSP 页 面 处 于 表现 层 , 也 就 是 视图 (View 即 V) 的 角色 。 

JavaBean 负责 数据 的 处 理 , 也 就 是 模型 (Model 即 M) 的 角色 。 

在 开发 中 ,采用 这 种 模式 ,可 以 具有 更 清晰 的 逻辑 划分 .能够 有 效 地 区 分 不 同 的 角色 ， 
避免 彼此 间 的 互相 影响 ,充分 发 挥 每 位 开发 人 员 的 特长 , 较 适 合用 于 开发 中 到 大 型 项 目 。 


课 后 练 习 


1. 将 “ 佳 衣 屋 ”项 目 实现 角色 权限 控制 功能 补充 完整 。 

提示 : 参考 示例 8-20, 完 成 总 店 店员 过 滤器 HeadFilter. java 和 分 店 店员 过 滤器 
BranchFilter 的 编写 ;按照 示例 8-21 的 Web. xml 中 的 配置 ,合理 组 织 JSP 文件 的 目录 结 
构 ,将 各 角色 的 受 控 使 用 JSP 文件 置 于 正确 的 目录 中 。 

2. 将 第 5 一 7 章 中 的 各 个 任务 用 JSP 十 JavaBean 十 Servlet 的 模式 来 改写 。 

3. 本 章 主 要 学 习 内 容 为 使 用 Servlet 和 Filter 技术 进行 权限 相关 的 安全 控制 。 安 全 
性 设计 和 实现 对 于 Web 应 用 的 正常 运行 至 关 重 要 ! 请 读者 结合 第 1 章 课 后 练习 的 要 求 
完成 “自选 项 目 需 求 分 析 ” 中 账号 安全 及 权限 控制 等 功能 。 
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本 章 学 习 要 点 : 

。 熟悉 Web 项 目 开发 常见 的 安全 性 设计 。 

。 熟练 掌握 使 用 JavaScript 进行 输入 的 合法 性 检查 。 
。 熟练 掌握 正则 表达 式 的 用 法 。 

。 熟练 掌握 Java 监听 器 编写 和 部 署 的 方法 。 


第 8 章 对 Web 项 目 中 与 账号 安全 性 控制 相关 的 技术 做 了 综合 讲述 ,但 Web 项 目的 
安全 性 设计 不 仅 限于 对 账号 的 控制 ,本 章 将 探讨 其 他 需要 考虑 到 的 安全 性 设计 及 其 实现 
技术 。 


9.1 任务 一 : 对 录入 的 员工 信息 做 合法 性 检查 


Web 应 用 中 有 许多 信息 需要 用 户 从 网 上 提交 ,用 户 可 能 由 于 朴 忽 ,输入 了 一 些 格式 
不 正确 的 信息 ,或 者 漏 填 了 一 些 必 不 可 少 的 信息 ,比如 在 员工 信息 的 录入 时 ,年 龄 中 输入 
了 非 整数 的 字符 串 ,或 者 漏 填 了 账号 和 密码 ,如 果 不 加 检查 就 直接 将 用 户 输入 的 信息 存 人 
数据 库 中 ,会 造成 保存 的 信息 失去 使 用 价值 ,更 严重 的 是 还 会 给 Web 应 用 带 来 安全 隐患， 
所 以 对 于 用 户 的 输入 信息 一 定 要 进行 合法 性 检查 。 

“任务 目标 : 


在 录入 商品 信息 时 需要 对 用 户 输入 的 信息 进行 合法 性 的 检查 。 要 求 账号 、 密 码 、; 
; 真实 姓名 必 填 ,密码 必须 是 6 位 的 字母 数字 串 , 用 户 权限 和 所 属 分 店 必 选 。 

”技能 训练 : 

: 。 使 用 JavaScript 进行 用 户 输入 的 合法 性 检查 。 

。 正则 表达 式 的 使 用 。 


在 录入 商品 信息 时 需要 对 用 户 输入 的 信息 进行 合法 性 的 检查 ,可 以 使 用 JavaScript 
技术 来 完成 ,如 示例 9-1 所 示 。 
兴 示 例 9-1 带 用 户 输入 合法 性 检查 的 员工 信息 录入 页 addEmployeeForm. jsp 


<%@page language="java" import="java.util.* 
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contentType="text/html;charset=gb18030"%> 


<center> 
员工 信息 录入 
<formname="forml" method="post" action="addEmployee.jsp" 
onsubmit="javascript: return addCheck()"> 
用 户 名 
<input type= "text" name="account"> 
<br> 
密码 
<input type="password" name= "password"> 
<br> 
姓名 
<input type="text" name="name"> 
<br> 
所 属 分 店 
<select name="storenum"> 
<option value="x"> 
请 选择 所 属 分 店 
</option> 
<option value="0"> 
总 店 
</option> 
<option value="1"> 
一 分 店 
</option> 
<option value="2"> 
二 分 店 
</option> 
</select> 
<br> 
授权 
<select name="authority"> 
> 


<option value= 
请 选择 用 户 权 限 
</option> 
<option value="0"> 
管理 员 
</option> 
<option value="1"> 
总 店 店员 
</option> 
<option value="2"> 
分 店 店员 
</option> 
</select> 
<br> 
<input type="submit" name="submit" value=" 添 加 " /> 
<input type="reset" name="reset" value=" 重 置 " /> 
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</form> 


</center> 


<script> 

function addCheck() { 
Var account=document .forml .account .value; 
Var password=document .forml .password.value; 


var name=document .forml .name.value; = 获取 表单 元 素 的 值 


Var storenum=document .forml.storenum.value; 


Var storenum=document .forml.authority.value; 


ee 当 检查 到 用 户 没有 填写 账号 时 ， 弹 出 
， 由; 没有 填写 账号 时 ， 
alert ("请 输入 账号 1") ;一 一 敬告 杠 ， 提 柄 用 户 账号 为 必 填 项 


将 光标 移 至 账号 元 素 内 闪烁 ， 帮 助 


document .forml .account .focus () je 用 户 快速 定位 出 现 问题 的 地 方 


return false; 一 二 返 


false， 表 单数 据 并 不 会 被 提交 | 


} 


var Regx=/^[A-2a-z0-9] {6}$/; = 定义 一 个 代表 6 位 字母 、 数 字 串 的 正则 表达 式 


if (!Regx.test (password)) { 用 这 个 正则 表达 式 检 查 用 户 输入 的 密码 


串 ， 则 不 提交 表单 数据 ， 并 要 


ore MA itis te A 
求 用 户 重新 填写 密码 


document .forml .password.focus (); 


return false; 

} 

if (name=="") { 
alert ("请 输入 真实 姓名 !1") ; 
document .forml .name.focus (); 
return false; 

} 

if (!(/^[0-9]{1}$/.test(storenum))) { 
alert ("请 选择 所 属 分 店 !1"); 
document .forml .storenum.focus () 7? 
return false; 

} 

if (!(/^[0-9]{1}$/.test(authority))) { 
alert (" 请 选择 用 户 权限 !") ; 
document .forml.authority.focus () 
return false; 


} 
</script> 
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9.2 任务 二 : 过 滤 用 户 的 恶意 输入 


Web 应 用 中 ,时 常会 遇 到 允许 用 户 输入 大 段 的 文字 ,例如 客户 留言 .对 商品 的 评价 
等 ,如 果 某 些 网 络 用 户 借 此 机 会 恶意 输入 一 段 HTML 标签 或 者 JS、JSP 代码 ,被 不 加 处 
理 地 直接 存储 在 数据 库 中 ,那么 在 这 些 信息 被 读 取 及 显示 的 时 候 , 会 被 当 作 有 效 代码 来 执 
行 ,而 不 是 仅仅 作为 内 容 来 输出 ,例如 以 下 代码 被 当 作用 户 留 言 存储 在 数据 库 中 , 当 打 开 
留言 信息 页 时 ,会 弹出 一 个 显示 “哈哈 1” 的 警告 框 。 

<script> 


alert(" 险 险 !"); 
<script> 


| 将 用 户 输入 信息 中 的 敏感 的 HTML 字符 转换 为 HTML 转 义 符 。 
”技能 训练 : 
: 。 Java 字符 串 的 处 理 。 
。 Web 应 用 安全 性 设计 意识 的 提高 。 

HTML 标签 JavaScript 程序 块 和 JSP 语法 的 构成 都 少不了 ”二 ”和 "> 这 两 个 符 
号 ,如 果 把 此 类 敏感 的 字符 转换 成 “&lt;” 和 "&-gt; "这 样 的 转 义 符 ,代码 语法 就 会 被 破坏 ， 
不 会 再 被 执行 了 。 下 面 编写 一 个 字符 转换 器 类 ,来 完成 本 章 任务 二 的 要 求 。 

闪 示 例 9-2 ”完成 HTML 转 义 符 转 换 的 ConvertorUtil. java 


package com.jyw.www.util; 


import java.text.*; 
import java.util.Date; 
import java.io.*; 


public class ConvertorUtil 
{ 


public static String convertToHtml (String s) 

{ 
s=replace(s, "&", "&amp;"); 
s=replace (s, "<", "glt;"); 
s=replace (s, ">", "ggt;"); 
s=replace (s, "\t", " 六 
s=replace (s, "\r\n", "\n"); = 将 一 些 敏感 字符 替换 成 HTML 转 义 符 
s=replace (s, "\n", "<br>"); 
s=replace (s, " "," &nbsp;"); 
s=replace (s, "'", "&#39;"); 
s=replace (s, "\\", "&#92;"); 
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return s; 


} 


public static String replace (String s, String sl, String s2) 


{ 
if(s==null) 
{ 
return null; 
} 


StringBuffer stringbuffer=new StringBuffer(); 

int i=s.length(); 

int j=sl.length(); 

int k; 

int 1; 

for (k=0; (l=s.indexOf (sl, k))>=0; k=1+j) 

{ 
stringbuffer.append(s.substring (k, 1)); 
stringbuffer.append(s2); 

} 


if (k<i) 
{ 

stringbuffer.append(s.substring (kK)); 
} 


return stringbuffer .toString (); 


} 


可 能 包含 敏感 字符 的 数据 在 存 和 数据库 之 前 ,调用 示例 9-2 中 的 convertToHtml() 
方法 可 过 滤 恶 意 的 输入 。 


9.3 任务 三 : 配置 首页 和 全 局 错误 提示 页 面 


读者 应 该 有 这 样 的 开发 体验 , 即 当 发 生 页 面 找 不 到 或 者 语法 错时 ,会 出 现 JSP 错误 
提示 信息 ,给 出 错误 发 生 的 位 置 和 原因 ,其 目的 是 帮助 开发 人 员 进 行 错误 定位 和 调试 。 但 
是 对 于 一 个 上 线 运行 的 Web 应 用 仍然 显示 JSP 的 错误 提示 信息 ,会 因为 暴露 系统 的 数据 
库 .服务 器 文件 路 径 等 信息 给 应 用 带 来 安全 隐患 。 这 时 ,需要 为 应 用 配置 一 个 安全 的 错 
误 提示 页 。 
”任务 目标 : 
”Web 应 用 系统 发 生 404 错误 时 , 跳 转 到 提示 “网 页 正在 建设 中 ...” 的 提示 页 ;Web 
| 应 用 系统 发 生 500 错误 时 , 跳 转 到 提示 “出 错 了 ,请 通知 系统 管理 员 ” 的 提示 页 。 : 
”技能 训练 : 
: Web. xml 中 全 局 错误 页 的 配置 。 
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在 Web. xml 中 配置 全 局 的 错误 提示 页 面 ,代码 如 下 。 
XA 示例 9-3 Web. xml 中 全 局 错误 提示 页 的 配置 


<error-page> 
<error-code>404< /error- code> 
<location>/building.jsp</location> 
</error-page> 
<error-page> 
<error-code>500</error- code> 
<location>/error.jsp</location> 
</error-page> 
错误 提示 页 的 编写 要 设置 page 指令 的 isErrorPage 属性 为 true, 表 示 当 前 页 可 以 作 
为 其 他 JSP 页 面 的 错误 页 面 ,并 设置 response 的 状态 为 “请 求 成 功 ”, 并 中 止 错误 状态 的 
延续 ,示例 如 下 : 
<%@page language="java" contentType="text/html; charset=GBK" 
isErrorPage="true" pageEncoding= "GBK"%> 
<%response.setSstatus (HttpServletResponse.sC OK);%> 


错误 提示 页 的 内 容 部 分 请 读者 自行 完成 。 
9.4 任务 四 : 统计 系统 在 线 人 数 


许多 Web 应 用 中 会 有 统计 当前 在 线 人 数 的 功能 ,使 得 用 户 或 者 管理 员 对 该 Web 应 
用 的 使 用 情况 有 一 个 直观 的 了 解 ,这 个 功能 可 以 通过 HttpSessionListener 监听 器 来 实 
现 。Session 监听 会 话 的 创建 或 者 销毁 , 它 实 现 HttpSessionListener 接口 ,并 且 实 现 接口 
的 两 个 方法 。 
。 void sessionCreated (HttpSessionEvent hse): 当 一 个 HttpSession 对 象 被 创建 
时 ,将 会 调用 这 个 方法 ; 
。 void sessionDestroyed( HttpSessionEvent hse) : 当 一 个 HttpSession 超时 或 者 调 
用 HttpSession 的 invalidate() 方 法 让 它 销毁 时 ,将 会 调用 这 个 方法 。 
示例 9-4 实现 了 一 个 简单 的 在 线 人 数 统计 。 
闪 示 例 9-4 ActiveUserListener. java 实现 在 线 人 数 统计 


package com.jyw.www.listener; 


import javax.servlet.*; 


import javax.servlet.http.*; 


public class ActiveUserListener implements HttpSessionListener { 


private static int sessionCount=0; 


public void sessionCreated(HttpSessionEvent se) { 
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sessionCount++; 


} 


public void sessionDestroyed(HttpSessionEvent se) { 
if (sessionCount>0) 
sessionCount--; 
} 
public static int getActiveUsersCount () 
{ 
return sessionCount; 


} 
} 


同 Servlet 和 Filter 一 样 ,Listener 也 需要 在 Web. xml 中 进行 部 署 ,如 示例 9-5 所 示 。 
关 示 例 9-5 Web. xml 中 监听 器 的 部 署 


<listener> 
<listener-class>com.mysite.www.listener.Listener 
</listener-class> 
</listener> 


9.5 知识 扩展 


下 面 介绍 一 下 正则 表达 式 。 

一 个 正则 表达 式 (Regular Expression) 就 是 由 普通 字符 (例如 字符 a 一 2 以 及 特殊 字 
符 ( 称 为 元 字符 ?组 成 的 文字 模式 。 该 模式 描述 了 在 查找 文字 主体 时 待 匹配 的 一 个 或 
多 个 字符 串 。 正 则 表达 式 作为 一 个 模板 ,将 某 个 字符 模式 与 所 搜索 的 字符 串 进行 匹 
配 。 构 建 正 则 表达 式 可 以 通过 查 表 来 完成 , 表 9-1 是 取 自 百 度 百科 词 条 “正则 表达 式 ” 
中 常用 的 元 字符 及 其 在 正则 表达 式 上 下 文中 的 行为 ,更 多 的 元 字符 含义 读者 请 自行 查 
找 资 料 。 


表 9-1 常用 的 元 字符 及 其 在 正则 表达 式 上 下 文中 的 行为 


字符 描 述 
\ 将 下 一 个 字符 标记 为 一 个 特殊 字符 ,或 一 个 原 义 字符 ,或 一 个 后 向 引用 ,或 一 个 八进制 转 
义 符 。 例 如 ,匹配 字符 "n"。ANan 匹 配 一 个 换行 符 。 序 列 \\ 匹 配 "\", 而 "\(" 匹配 "(” 
关 匹配 前 面 的 子 表达 式 零 次 或 多 次 。 例 如 ,zo * 能 匹配 "z" 以 及 "zoo"。* 等 价 于 {0,} 
匹配 前 面 的 子 表达 式 一 次 或 多 次 。 例 如 ,zo 十 能 匹配 "zo" 以 及 "zoo" ,但 不 能 匹配 "z"。 十 
等 价 于 {1,} 
匹配 前 面 的 子 表达 式 零 次 或 一 次 。 例 如 ,"do(es)?" 可 以 匹配 "do" 或 "does" 中 的 "do"。? 等 
. 价 于 {0,1) 
pe n 是 一 个 非 负 整 数 。 匹 配 确 定 的 n 次 。 例 如 ,'o12)' 不 能 匹配 "Bob" 中 的 'o', 但 是 能 匹配 
"food" 中 的 两 个 o 


187 


中 小 型 Web 项 目 开 发 实战 


续 表 
字符 描 述 

人 n 是 一 个 非 负 整数 。 至 少 匹 配 n 次。 例如 ,'o{2,}' 不 能 匹配 "Bob" 中 的 'o', 但 能 匹配 

on "foooood" 中 的 所 有 o。'ofl,}) 等 价 于 o 十 '。'of0,) 则 等 价 于 ox* 
m 和 n 均 为 非 负 整数 ,其 中 n 一 二 m。 最 少 匹 配 n 次 且 最 多 匹配 m 次 。 例 如 :"o{1,3}" 将 

{n,m)} 匹配 "fooooood" 中 的 前 三 个 o。'o{0,1)' 等 价 于 '0?'。 请 注意 在 逗号 和 两 个 数 之 间 不 能 有 
空格 

zly 匹配 x 或 y。 例 如 ,'z|food' 能 匹配 "z" 或 "foo0d"。"(z| 人 fo0d' 则 匹配 "zood" 或 "food" 

[zyz] 字符 集合 。 匹 配 所 包含 的 任意 一 个 字符 。 例 如 ,[abc] 可 以 匹配 "plain" 中 的 

[zyz] 负 值 字符 集合 。 匹 配 未 包含 的 任意 字符 。 例 如 ,[^abcj] 可 以 匹配 "plain" 中 的 'p' 

区 三 志 字符 范围 。 匹 配 指定 范围 内 的 任意 字符 。 例 如 ,La 一 z]J] 可 以 匹配 a 到 % 范 围 内 的 任意 的 小 

写字 母 字 符 
区 负 值 字符 范围 。 匹 配 任何 不 在 指定 范围 内 的 任意 字符 。 例 如 ,La 一 zj 可 以 匹配 任何 不 在 
”| a 到 % 范 围 内 的 任意 字符 

Nb 匹配 一 个 单词 边界 ,也 就 是 指 单词 和 空格 间 的 位 置 。 例 如 ,erb 可 以 匹配 "never" 中 的 'er， 
但 不 能 匹配 "verb" 中 的 'er' 

\B 匹配 非 单词 边界 。'erB 能 匹配 "verb" 中 的 'er, 但 不 能 匹配 "never" 中 的 'er 

匹配 由 x 指明 的 控制 字符 。 例 如 ,\cM 匹配 一 个 ControLM 或 回 车 符 。x 的 值 必须 为 A 一 
Z 或 a~z 之 一 。 否则 ,将 c 视 为 一 个 原 义 的 'c 字 符 

\d 匹配 一 个 数字 字符 ,等 价 于 [0 一 9] 

\D 匹配 一 个 非 数 字 字 符 ,等 价 于 [^0 一 9] 

\f 匹配 一 个 换 页 符 , 等 价 于 \x0c 和 \cL 

\n 匹配 一 个 换行 符 ,等 价 于 \x0a 和 \cJ 

\r 匹配 一 个 回 等 价 于 \x0d 和 \cM 

\s 匹配 任何 空白 字符 ,包括 空格 、 制 表 符 、 换 页 符 等 。 等 价 于 [\f\n\r\t\v] 

XS 匹配 任何 非 空白 字符 ,等 价 于 [\fAnNrtAv] 

Nt 匹配 一 个 制 表 符 ,等 价 于 \x09 和 \cI 

\v 匹配 一 个 垂直 制 表 符 ,等 价 于 \x0b 和 \cK 

\w 匹配 包括 下 面 线 的 任何 单词 字符 ,等 价 于 [A 一 Za 一 z0 一 9_ 

\W 匹配 任何 非 单词 字符 ,等 价 于 [^A 一 Za 一 z0 一 9_]' 


课 后 练习 


1. 为 " 佳 衣 屋 "的 任何 来 自用 户 的 数据 添加 合法 性 安全 控制 。 
2. 将 本 章 讲授 的 安全 性 设计 运用 到 "自选 项 目 "中 。 
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本 章 学 习 要 点 : 

。 熟练 掌握 将 Web 数据 导出 到 Excel 中 的 方法 。 
。 熟练 掌握 将 Web 数据 导出 到 Word 中 的 方法 。 
。 熟练 掌握 使 用 JFreeChart 绘制 柱 图 的 方法 。 

。 熟练 掌握 使 用 JFreeChart 绘制 饼 图 的 方法 。 

。 熟练 掌握 使 用 JFreeChart 绘制 折线 图 的 方法 。 


经 过 前 三 部 分 的 建设 后 ,“ 佳 衣 屋 "信息 管理 系统 的 日 常 管理 功能 基本 实现 , 随 着 系统 
的 上 线 运 行 , 因 和 人 库 .销售 等 行为 ,在 数据 库 中 ”收集 "了 许多 经 营 数据 ,通过 对 经 营 数据 的 
分 析 ,企业 管理 者 可 以 及 时 了 解 经 营 现状 ,发 现 问题 ,辅助 未 来 经 营 管理 的 决策 。 


10.1 任务 一 : 将 数据 导出 至 Excel 中 


Microsoft Excel 是 微软 公司 Microsoft Office 办 公 软 件 的 组 件 之 一 , 它 可 以 进行 各 种 
数据 的 处 理 、 统 计 分 析 和 辅助 决策 操作 ,广泛 地 应 用 于 管理 .统计 财经 .金融 等 众多 领域 ， 
可 以 将 数据 表 中 的 信息 导出 到 Excel 中 ,再 使 用 Excel 的 内 置 功 能 完成 数据 分 析 。 
“任务 目标 : 

”将 各 分 店 的 调 货 请 求 信息 导出 到 Excel 表 中 。 
”技能 训练 : 
;视图 的 创建 。 

可 以 通过 设置 MIME 类 型 为 来 完成 任务 一 的 要 求 。MIME 类 型 就 是 设 定 某 种 扩展 
名 的 文件 用 一 种 应 用 程序 来 打开 的 方式 类 型 , 当 该 扩展 名 文件 被 访问 的 时 候 , 浏 览 器 会 自 
动 使 用 指定 应 用 程序 来 打开 。 在 JSP 文件 中 通过 设置 page 指令 的 contentType 属性 的 
值 来 指定 MIME 类 型 ,用 Excel 打开 的 . xls 文件 的 MIME 类 型 为 “application/msexcel” 
如 示例 10-1 所 示 。 


nd Ses dle nde 


闪 示 例 10-1 showExcel.jsp 


<%@page language="java" import="java.util.*" pageEncoding= "utf-8" $> 


<%@page import="java.util.* ,com.jyw.www.dao.* ,com.jyw.www.entity.*" %> 
<%@page contentType="application/msexcel" %> 


本 设置 标题 字体 


response.setHeader ("Content-disposition","attachment; filename=showExcel .xls"); 


$%> 
<center> 商 品 调 货 情况 </center> 
<table border=1 align="center" bgColor="#ffffff"> 
<tr> 
<td align=center><B> 序 号 </B></td> 
<td align=center><B> 款 号 </B></td> 
<td align=center><B> 颜 色 </B></td> 
<td align=center><B> 类 别 </B></td> 
<td align= center><B> 码 数 </B></td> 
<td align=center><B> 数 量 </B></td> 
<td align=center><B> 调 货 日 期 </B>< /td> 
<td align=center><B> 处 理 日 期 </B></td> 
<td align=center><B> 状 态 </B></td> 
</tr> 
<% 
OrderinfoDAO orderinfoDao=new OrderinfoDAO(); 
List<Orderinfo>orderinfoList=orderinfoDao.findAll (); 


设置 输出 .xls 文 件 的 名 字 


for (Orderinfo orderinfo:orderinfoList) { 

多 > 

<tr> 
<td align=center><%=orderinfo.getId()%></td> 
<td align=center><%=orderinfo.getGoodsid()%></td> 
<td align=center><$%$=orderinfo.getColor ()$></td> 
<td align=center><$=orderinfo.getName ()%></td> 
<td align=center><%=orderinfo.getSize()%></td> 
<td align=center><%=orderinfo.getQuantity()®%></td> 
<td align=center><$%=orderinfo.getApplydate()%></td> 
<td align=center> 


<%=orderinfo.getProcessdate()!=null?orderinfo.getProcessdate():""$%> 
</td> 
<td align=center><%s=orderinfo.getOrderstatus ()%>< /td> 
<% 
} 
%> 
</tr> 
</table> 


运行 showExcel. jsp 时 会 弹出 一 个 窗口 ,询问 用 户 是 “保存 ”还 是 直接 “打开 ”此 xls 文 
件 , 任 何 一 种 方式 下 .都 可 以 得 到 如 图 10-1 所 示 的 商品 调 货 情况 的 Excel 文件 。 


192 


第 10 章 图 表 的 生成 


颜色 | 类 别 调 货 日 期 | 处 理 日 期 


水 洗 黑 | 牛仔裤 2013/2/22 | 2013/2/22 
水 洗 黑 | 牛仔裤 2013/2/22 | 2013/4/5 
水 洗 黑 | 牛仔裤 2013/3/31 | 2013/4/5 
蓝 | 牛仔 裤 2013/3/31 


10-1 从 Web 导出 的 Excel 文件 


10.2 任务 二 : 将 数据 输出 至 Word 中 


微软 公司 Office 系列 办 公 组 件 中 的 另 一 个 软件 Word, 是 目前 世界 上 最 流行 的 文字 
编辑 软件 ,拥有 强大 的 图 片 混 排 和 表格 制作 功能 。Web 数据 也 常常 因为 存档 及 再 次 编 
辑 、 打 印 等 原因 需要 导出 到 Word 中 。 


“任务 目标 : 

”将 各 分 店 的 调 货 请 求 信息 导出 到 Word 文件 中 。 
”技能 训练 : 

”视图 的 创建 


与 任务 一 相似 ,将 MIME 类 型 设置 为 application/msword, 就 可 以 将 Web 数据 列表 
导出 到 Word 中 了 ,如 图 10-2 所 示 。 


[二 -9 showWord doc - Microsof Word E 
ee 


商品 调 货 情况 " 

款 号 “| 颜色 | 类 别 码 数 数量 | 订购 日 期 | 处理 日 期 | 状态” 
JYW12CENL 水 洗 黑 秆 仔裤 | S。| 10 [2013-02-22|2013-02-22 | 待 发 货 
[TYR12CEN1 水 洗 黑 慎 仔 裤 | Me | 20e |2013-02-22|2013-04-05 已 发 货 
[TYW12CENL 水 洗 黑 牛仔 裤 | Me | 2。 |2013-03-31|2013-04-05 已 取消 
Txwl2cKB0| 蓝 。 目 仔裤 | Se | 2。 |2013-03-31|e F 


图 10-2 从 Web 导出 的 Word 文件 


关 示 例 10-2 showWord.jsp 中 指定 MIME 类 型 和 导出 文件 名 的 语句 


<$sQ@page contentType="application/msword" gs> 
< 
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response.setHeader ("Content-disposition", "attachment;filename=showWord.doc"); 


10.3 任务 三 : 各 分 店 销量 的 柱 图 统计 


表格 列 出 的 只 是 未 经 处 理 和 分 析 的 原始 数据 ,不 像 图 形 图 像 的 数据 表示 形式 那样 可 
以 直观 地 反映 数据 中 蕴藏 的 变化 和 规律 。JFreeChart 是 一 组 功能 强大 、 灵 活 易 用 的 Java 
绘图 第 三 方 组 件 。 在 Web 中 得 到 广泛 的 应 用 ,使 用 它 可 以 生成 多 种 报表 ,包括 柱状 图 、 饼 
图 ,折线 图 等 。 

柱 形 图 是 将 一 个 或 多 个 类 别 的 数据 进行 可 视 化 的 好 方法 ,用 来 显示 一 段 时 间 内 数据 
的 变化 或 者 描述 各 项 目 之 间 的 数据 比较 ,直观 地 表现 出 各 类 数据 点 的 差别 , 它 是 最 常用 的 
图 表 类 型 之 一 。 


”任务 目标 : 

为 佳 衣 屋 项 目 添加 的 月 销售 量 统 计 柱 图 。 
”技能 训练 : 

| 。 视图 的 创建 。 


。 实体 类 的 编写 规范 。 
。 DAO 类 中 findAll() 方 法 的 编写 规范 。 
。 使 用 JFreeChart 绘制 柱 图 。 


第 3 章 任务 四 中 ,创建 了 视图 salesinfo, 如 图 10-3 所 示 ,这 个 视图 中 记录 了 所 发 生 的 
每 一 笔 交 易 ,月 销量 可 以 根据 此 视图 来 进行 统计 。 


她 ! 条 信息 | 回 2 条 表 数 据 


辣 皮 " 吕 局 鲁 介 网 玫 才 属 cP | 万 限 行 第 - 行 ， .JP 十 行 和 匠 | | 


图 10-3 视图 salesinfo 中 的 记录 


再 创建 一 个 视图 salesstatistic, 脚 本 如 示例 10-3 所 示 ,这 个 视图 将 按 月 统计 销售 量 和 
销售 额 。 
闪 示 例 10-3 ”视图 salesstatistic 的 SQL 脚本 


DELIMITER $$ 
USE 'jyw'$$ 
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DROP VIEW IF EXISTS 'salesstatistic'$$ 


CREATE ALGORITHM=UNDEFINED DEFINER= 'root'@ 'localhost" SQL SECURITY 
DEFINER VIEW "salesstatistic' RS 
SELECT 
DATE FORMAT('salesinfo'.'date','%Y- $m') RS 'month', 
SUM('salesinfo'.'quantity') RS 'quantity', 
SUM(('salesinfo'.'sellingprice'* 'salesinfo'.'quantity')) AS 'sales' 
FROM 'salesinfo' 
GROUP BY DATE FORMAT('salesinfo'.'date','%Y- sm'") 
ORDER BY DATE FORMAT('salesinfo'.'date','%Y- %m')$$ 


DELIMITER; 


salesstatistic 视图 创建 完成 后 ,其 中 的 记录 如 图 10-4 所 示 。 

按照 第 5 章 中 讲授 的 方法 ,为 salesstatistic 视图 创建 相应 
的 实体 Bean SalesStatistic 和 DAO 类 SalesStatisticDAO , 并 为 
DAO 类 添加 findAll() 方 法 ,将 视图 中 的 相关 数据 读 出 后 就 可 
以 利用 JFreeChart 绘制 图 表 了 。 

使 用 JFreeChart, 需要 在 Web. xml 文件 中 增加 如 下 配 
置 ,参考 示例 10-4。 

闪 示 例 10-4 Web. xml 中 JFreeChart 的 配置 


图 10-4 视图 salesstatistic 


<servlet> 中 的 记录 
<servlet-name>DisplayChart</servlet-name> 


<servlet-class>org.jfree.chart .servlet.DisplayChart</servlet-class> 
</servlet> 
<servlet-mapping> 

<servlet-name>DisplayChart</servlet-name> 

<url-pattern> /DisplayChart< /url-pattern> 
</servlet-mapping> 


下 面 编写 月 销量 统计 的 柱 形 图 的 显示 文件 bar. jsp。 
闪 示 例 10-5 bar. jsp 


<%@page contentType= "text/html;charset=GB18030"%> 
<%@page 
import="org.jfree.chart.*, 
org.jfree.chart .plot.*, 
org.jfree.chart.servlet.ServletUtilities, 
org.jfree.chart.title.x*, 
org.jfree.data.category.DefaultCategoryDataset, 
java.awt.Font, | 引入 JFreeChart 有 包 
org.jfree.data.category.CategoryDataset, 
org.jfree.data.general .DatasetUtilities, 
org.jfree.chart .axis.*, 


javax.naming.InitialContext, 


javax.sql .DataSource, 
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上 


java.sql. *, 
Com.jyw.www.dao.*, 
java. util,.*,; 
Com.jyw.www.entity. 关 ™ 


多 > 
<% 
DefaultCategoryDataset dataset=new DefaultCategoryDataset (); 
SalesStatisticDAO salesStatisticDAO=new SalesStatisticDAO(); 
List<SalesStatistic>salesList=salesStatisticDAO.findAll (); 
for(SalesStatistic salesstatistic: salesList) { 
dataset .addValue (salesstatistic.getQuantity(),"",salesstatistic.getMonth()); 
} 将 月 份 和 销售 量 信息 添加 到 数据 集中 


JFreeChart chart=ChartFactory.createBarChart3D(" 销 量 量 统计 柱 图 "，""， 
"销量 ", dataset, PlotOrientation.VERTICAL, false, false, 
false); 


对 数据 集 应 用 柱 图 主题 样式 
TextTitle textTitle=chart.getTitle(); 


textTitle.setFont (new Font ("宋体 ", Font .BOLD, 20)) | 设置 标题 字体 


LegendTitle legend=chart .getLegend () 
if (legend !=null) { 
legend.setItemFont (new Font ("宋体 ",， Font .BOLD, 20)); 


} 设置 图 例 的 字体 
CategoryPlot plot= (CategoryPlot) chart .getPlot (); 
CategoryAxis domainAxis=plot.getDomainAxis(); 
设置 x 轴 上 的 字体 
domainAxis.setTickLabelFont (new Font (" 宋 体 "，Font .BOLD，20) ) 到 一 
设置 x 轴 上 标题 的 字体 


domainAxis.setLabelFont (new Font ("宋体 ", Font .BOLD, 20)); 


i 之 
Valueaxis valueAxis=plot .getRangeAxis () ;// 柱 形 图 的 y 轴 设置 y 轴 上 的 字体 
valueAxis.setTickLabelFont (new Font (" 宋 体 "，Font .BOLD, 20)); 


valueAxis.setLabelFont (new Font ("宋体 ", Font .BOLD, 20)); 
设置 y 轴 上 字体 


String filename=ServletUtilities.saveChartAsPNG (chart, 500, 300,null, session); 


String graphURL=request .getContextPath () 将 柱 图 统计 存储 为 png 文 件 
+"/DisplayChart? filename="+filename; 


%> 
<center> 获取 柱 图 的 图 片 文件 URL 


<img src="<%=graphURL%>" width=500 height=300 border=0 
usemap="#<%=filename %>"> 3 
| 显示 柱 图 图 片 


</center> 
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bar.jsp 的 执行 效果 如 图 10-5 所 示 。 


L=JDjx| 


GO - [Bhocahostaoe0/sdmin/barjsp -| [| |x| Es-T. sn 四 日 


窗 变 镍 http//ocalhost8080/admin/barj.. 从 "上 国 了 各 了 上 PD) 7 使 IRO)v” 


销售 量 统 计 柱 图 


图 
| | |@ intemnet| SP 模式 启用 [R100% > 


图 10-5 月 销量 统计 柱 图 


10.4 任务 四 : 商品 销售 额 折 线 图 


折线 图 是 将 一 系列 数据 连接 起 来 ,可 以 表现 数据 的 动态 变化 。 


”任务 目标 : 
| 为 佳 衣 屋 项 目 添 加 月 销售 额 统计 折线 图 。 
技能 训练， 


使 用 JFreeChart 绘制 折线 图 。 


下 面 编写 月 销售 额 统计 的 折线 图 的 显示 文件 salesline.jsp.jsp。 
闪 示 例 10-6 salesline. jsp 


<%$@page contentType="text/html;charset=GB18030"%> 

<%@page 
import="org.jfree.chart.*, 
org.jfree.chart.plot.*, 
org.jfree.chart.servlet.ServletUtilities, 
org.jfree.chart.title.*, 
org.jfree.data.category.DefaultCategoryDataset, 
java.awt .Font, 
org.jfree.data.category.CategoryDataset, 
org.jfree.data.general .DatasetUtilities, 
org.jfree.chart.axis.*, 


javax.naming.InitialContext, 
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<% 


javax.sql.DataSource, 
java.sql.*, 
Com.jyw.www.dao.*, 
Com.jyw.www.entity.*, 


Com.jyw.www.util.* "%> 


Test ld=new Test (); 
List<String[]>list-new ArrayList<String[]> (); 


SalesStatisticDAO salesStatisticDao=new SalesStatisticDAO(); 
List<SalesStatistic>salesList=salesStatisticDao.findAll (); 
DefaultCategoryDataset dataset=new DefaultCategoryDataset (); 


for (SalesStatistic salesstatistic : salesList) { 


dataset .addValue (Double.valueOf (String.valueOf (salesstatistic. 


getSales(),"",salesstatistic.getMonth()); 
} 
st.setRegularFont (new Font ("宋体 "，Font .PLAIN, 15)); 


JFreeChart chart=ChartFactory.createLineChart (一 -一 


对 数据 集 应 用 
折线 图 主题 样式 


"销售 量 统计 折线 设置 标 是 

"月 份 "， = 
设置 x 轴 标题 

"销售 额 "， 设置 y 轴 标题 


dataset, // data 


PlotOrientation.VERTICAL, 一 一 一 显示 方向 


例 ，false 为 不 显示 


CategoryPlot line=chart.getCategoryPlot () 
TextTitle textTitle=chart .getTitle(); 
textTitle.setFont (new Font ("宋体 ", Font .BOLD, 20)); 
LegendTitle legend=chart .getLegend(); 

if (legend !=null) { 


legend.setItemFont (new Font ("宋体 ",，Font .BOLD, 20)); 


} 
CategoryPlot plot= (CategoryPlot) chart .getPlot (); 
CategoryAxis domainAxis=plot.getDomainAxis() 


domainAxis.setTickLabelFont (new Font ("宋体 ", Font .BOLD, 20)); 
domainAxis.setLabelFont (new Font ("宋体 ", Font .BOLD, 20)); 


ValueAxis valueAxis=plot.getRangeAxis (); 


valueAxis.setTickLabelFont (new Font ("宋体 ", Font .BOLD, 20)); 
valueAxis.setLabelFont (new Font ("宋体 ", Font .BOLD, 20)); 


filename=ServletUtilities.saveChartAsPNG (chart, 600, 300, 


null, request .getSession()); 
} catch (Exception e) { 
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e.printstackTrace () 7 
} 
String graphURL=request .getContextPath() 
+"/DisplayChart? filename="+filename; 
> 
<center> 
<img src='<%=graphURL%>" width= 600 height=300 border=0 
usemap= '#<%=filename®%>'> 
</center> 


salesline. jsp 的 执行 效果 如 图 10-6 所 示 。 


TO |B hp/ocahostso80/admin/salesinels +] [|[x| [Ee—F, sn Ea| 


窗 袍 ”起 htpy/localhostsoso/admin/salesl. 丛 ~ 园 ~ 咒 ~ 司 Xp 对 IRov” 


销售 额 统计 折线 图 


2013-02 2013-03 2013-04 2013-05 


10-6 月 销售 额 折 线 图 


10.5 任务 五 : 各 类 商品 销量 的 饼 图 统计 


饼 图 是 一 个 划分 为 几 个 扇形 的 圆 形 统计 图 表 , 用 于 描述 量 或 百分比 之 间 的 相对 关系 。 


”任务 目标 : 
: 为 佳 衣 屋 项 目 添加 销售 量 分 布 饼 图 。 
”技能 训练 : 
。 视图 的 创建 。 
。 实体 类 的 编写 规范 。 
。 DAO 类 中 findAll() 方 法 的 编写 规范 。 
。 使 用 JFreeChart 绘制 饼 图 。 


创建 一 个 视图 catigoriesStatistic, 脚 本 如 示例 10-7 所 示 , 这 个 视图 将 按 月 统计 销售 
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量 和 销售 额 。 


图 10-7 所 示 。 


创建 相应 的 实体 Bean CatigoriesStatistic 和 DAO 类 
CatigoriesStatisticDAO, 并 为 DAO 类 添加 findAll() 方 
法 ,将 视图 中 的 相关 数据 读 出 后 ,就 可 以 利用 JFreeChart 


闪 示 例 10-7 视图 catigoriesStatistic 的 SQL 脚本 
DELIMITER $$ 


USE 'jyw'$$ 
DROP VIEW IF EXISTS 'catigoriesSstatistic'$$ 
CREATE ALGORITHM= UNDEFINED DEFINER= ' root '@ ' localhost ' SQL SECURITY DEFINER 
VIEW 'catigoriesstatistic' RS 
SELECT 
"salesinfo'.'name' AS 'id', 
SUM('salesinfo'.'quantity') RS 'quantity' 
FROM 'salesinfo' 
GROUP BY 'salesinfo'.'name'$$ 
DELIMITER; 


catigoriesStatistic 视图 创建 完成 后 ,其 中 的 记录 如 


按照 第 5 章 中 讲授 的 方法 ,为 catigoriesStatistic 视图 


绘制 图 表 了 。 


图 10-7 视图 catigoriesStatistic 


pie.jsp 文件 中 用 于 显示 各 类 型 销量 统计 的 饼 形 图 的 


部 分 代码 如 下 。 
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闪 示 例 10-8 pie.jsp 


<%@page contentType="text/html;charset=GBK"%> 
<%@page 

import =" org. jfree. chart. * ， org. jfree. chart. plot. PiePlot, org. jfree. data. 

general. DefaultPieDataset, org. jfree. chart. servlet. ServletUtilities, 
org.jfree. chart. title. * , java.awt. Font, com. jyw. www. dao. * , com. jyw. 
www.entity.* ,java.util. * "%> 

< 多 

DefaultPieDataset dataset=new DefaultPieDataset (); 

CatigoriesStatisticDAO catigoriesStatisticDao=new CatigoriesStatisticDRO () 

List<CatigoriesStatistic>catigoriesStatisticList= 

catigoriesStatisticDao.findAll (); 

For (CatigoriesStatistic catigoriesStatistic: catigoriesStatisticList){ 
dataset .setValue (catigoriesStatistic.getId()， 
catigoriesStatistic.getQuantity()); 

} 

JFreeChart chart=ChartFactory.createPieChart3D ("各 类 型 销量 分 布 图 "， 

dataset, true, false, false); 


对 数据 集 应 用 饼 图 样式 


TextTitle textTitle=chart.getTitle(); 
textTitle.setFont (new Font ("宋体 ", Font .BOLD, 20)); 


第 10 章 图 表 的 生成 


LegendTitle legend=chart.getLegend () 
if (legend !=nul1l) { 
legend.setItemFont (new Font ("宋体 "， Font .BOLD, 20)); 
} 
PiePlot pieplot= (PiePlot) chart.getPlot(); 
pieplot.setLabelFont (new Font ("宋体 "， 0 12))7 
Font LegendFont=new Font ("楷体 ",， Font .PLAIN, 18); 
pieplot.setNoDataMessage ("无 数据 显示 "); 
pieplot.setCircular (false); 
pieplot.setLabelGap (0.02D); 
String filename=ServletUtilities.saveChartAsPNG (chart, 500, 300, 
null, session); 
String graphURL=request .getContextPath()+"/DisplayChart? 
filename="+filename; 
多 > 
<center> 
<img Src="<%=graphURLS>" width=500 height= 300 border=0 
usemap="#<%=filename $>"> 
</center> 


salesline. jsp 的 执行 效果 如 图 10-8 所 示 。 


TO [Brocahostaos0/admin/pielsp | [S|[x)| [下 ,AS FF 


宽 安 ”条 httpy/localhostsoso/adminrypiej,.- 丛 ~ 园 ~- 山 ~ 局 FEpv 对 IRov” 


格子 衬衣 。 牛 仔裤 卫衣 加 
4 


| [| [ | |@ intemet| St 启用 [Ei00% > 


图 10-8 ”分 类 销售 量 的 统计 饼 图 


课 后 练习 


1. 将 公司 员工 信息 导出 到 Excel 中 。 
要 求 : 结合 实际 应 用 ,添加 员工 工资 信息 表 salary ,将 员工 的 月 工资 导出 到 Excel 中 ， 
形成 工资 表 。 
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中 小 型 Web 项 目 开 发 实战 

提示 : 参考 本 章 任务 一 ,完成 本 练习 。 

2. 为 分 店 添加 分 店 销售 量 折线 图 统计 的 功能 。 

要 求 : 分 店 店员 登录 系统 后 ,可 查看 本 分 店 的 周 销售 量 和 销售 额 的 折线 图 。 

提示 : 参考 本 章 任务 四 ,完成 本 练习 。 

3. 为 管理 员 添加 分 店 销售 额 柱 图 。 

要 求 : 管理 员 登 录 系 统 后 ,可 查看 指定 月 份 的 各 分 店 销售 额 的 柱 图 。 

提示 : 参考 本 章 任务 三 ,完成 本 练习 。 

4. 本 章 主 要 学 习 内 容 为 数据 报表 的 生成 。 可 按照 第 一 章 课 后 练习 的 要 求 完 成 “自选 
项 目 需求 分 析 ” 项 目 中 应 有 的 数据 报表 分 析 。 
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中 
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